题意很简单

给一个树(n < 5w) 每个点有个权值,代表商品价格

若干个询问(5w)

对每个询问,问的是从u点走到v点(简单路径),商人在这个路径中的某点买入商品,然后在某点再卖出商品,   最大可能是多少

注意一条路径上只能买卖一次,先买才能卖

  • *分析:先求出点u,v的最近公共祖先f,然后求u->f->v的利润最大值maxval
  •  
    对于这个maxval可能有三种情况:
  •  
    1:maxval是u->f的maxval
  •  
    2:maxval是f->v的maxval
  •  
    3:maxval是u->f的最小w[i]减去f->v的最大w[i]
  •  
    分析到这很明显需要设置4个变量来求maxval:
  •  
    up[u]表示u->f的最大maxval
  •  
    down[u]表示f->u的最大maxval
  •  
    maxw[u]表示u-f的最大w[i]
  •  
    minw[u]表示u-f的最小w[i]
  •  
    所以maxval=max(max(up[u],down[v]),maxw[v]-minw[u]);
  •  
    现在问题就是如何快速的求出这四个变量,在这里我们可以对u,v的LCA(u,v)进行分类解决
  •  
    对于LCA(u,v)是f的询问全部求出,然后再求LCA(u,v)是f的父亲的询问
  •  
    这样当我们求LCA(u,v)是f的父亲的询问的时候就可以借用已经求出的LCA(u,v)是f的询问
  •  
    的结果,这样就不用反复去求u->f的那四个变量值,u->father[f]也能快速求出
  •  
    这个变化主要在寻找father[v]这个过程中进行,具体看代码
  • /*分析:先求出点u,v的最近公共祖先f,然后求u->f->v的利润最大值maxval
    对于这个maxval可能有三种情况:
    1:maxval是u->f的maxval
    2:maxval是f->v的maxval
    3:maxval是u->f的最小w[i]减去f->v的最大w[i]
    分析到这很明显需要设置4个变量来求maxval:
    up[u]表示u->f的最大maxval
    down[u]表示f->u的最大maxval
    maxw[u]表示u-f的最大w[i]
    minw[u]表示u-f的最小w[i]
    所以maxval=max(max(up[u],down[v]),maxw[v]-minw[u]);
    现在问题就是如何快速的求出这四个变量,在这里我们可以对u,v的LCA(u,v)进行分类解决
    对于LCA(u,v)是f的询问全部求出,然后再求LCA(u,v)是f的父亲的询问
    这样当我们求LCA(u,v)是f的父亲的询问的时候就可以借用已经求出的LCA(u,v)是f的询问
    的结果,这样就不用反复去求u->f的那四个变量值,u->father[f]也能快速求出
    这个变化主要在寻找father[v]这个过程中进行,具体看代码
    */
    #include <iostream>
    #include <cstdio>
    #include <cstdlib>
    #include <cstring>
    #include <string>
    #include <queue>
    #include <algorithm>
    #include <map>
    #include <cmath>
    #include <iomanip>
    #define INF 99999999
    typedef long long LL;
    using namespace std; const int MAX=+;
    int n,m,size;
    int uu[MAX],vv[MAX],ww[MAX],sum[MAX];
    int up[MAX],down[MAX],maxw[MAX],minw[MAX],father[MAX];
    int head[MAX],head2[MAX],head3[MAX];
    bool mark[MAX]; struct Edge{
    int v,id,next;
    Edge(){}
    Edge(int V,int ID,int NEXT):v(V),id(ID),next(NEXT){}
    }edge[MAX*],edge2[MAX*],edge3[MAX*]; void Init(int num){
    for(int i=;i<=num;++i)head[i]=head2[i]=head3[i]=-,mark[i]=false;
    size=;
    }
    void InsertEdge(int u,int v,int id){
    edge[size]=Edge(v,id,head[u]);
    head[u]=size++;
    }
    void InsertEdge2(int u,int v,int id){
    edge2[size]=Edge(v,id,head2[u]);
    head2[u]=size++;
    }
    void InsertEdge3(int u,int v,int id){
    edge3[size]=Edge(v,id,head3[u]);
    head3[u]=size++;
    }
    int findset(int v){
    if(v == father[v])return father[v];
    int fa=father[v];
    father[v]=findset(father[v]);
    up[v]=max(max(up[v],up[fa]),maxw[fa]-minw[v]);
    down[v]=max(max(down[v],down[fa]),maxw[v]-minw[fa]);
    maxw[v]=max(maxw[v],maxw[fa]);
    minw[v]=min(minw[v],minw[fa]);
    return father[v];
    }
    void LCA(int u){
    mark[u]=true;
    father[u]=u;
    for(int i=head2[u];i != -;i=edge2[i].next){//对LCA(u,v)进行分类
    int v=edge2[i].v,id=edge2[i].id;
    if(!mark[v])continue;
    int f=findset(v);
    InsertEdge3(f,v,id);
    }
    for(int i=head[u];i != -;i=edge[i].next){
    int v=edge[i].v;
    if(mark[v])continue;
    LCA(v);
    father[v]=u;
    }
    for(int i=head3[u];i != -;i=edge3[i].next){
    int id=edge3[i].id;
    findset(uu[id]);
    findset(vv[id]);
    sum[id]=max(max(up[uu[id]],down[vv[id]]),maxw[vv[id]]-minw[uu[id]]);
    }
    }
    int main(){
    int u,v;
    while(~scanf("%d",&n)){
    Init(n);
    for(int i=;i<=n;++i){
    scanf("%d",ww+i);
    up[i]=down[i]=;
    maxw[i]=minw[i]=ww[i];
    }
    for(int i=;i<n;++i){
    scanf("%d%d",&u,&v);
    InsertEdge(u,v,i);
    InsertEdge(v,u,i);
    }
    size=;
    scanf("%d",&m);
    for(int i=;i<m;++i){
    scanf("%d%d",&uu[i],&vv[i]);
    InsertEdge2(uu[i],vv[i],i);
    InsertEdge2(vv[i],uu[i],i);
    }
    size=;
    LCA();
    for(int i=;i<m;++i)printf("%d\n",sum[i]);
    }
    return ;
    }

RMQ做法

#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <string>
#include <queue>
#include <algorithm>
#include <map>
#include <cmath>
#include <iomanip>
#define INF 99999999
typedef long long LL;
using namespace std; const int MAX=+;
int n,m,size,top;
int uu[MAX],vv[MAX],ww[MAX],anc[MAX];
int up[MAX][],down[MAX][],maxw[MAX][],minw[MAX][],deep[MAX];
int head[MAX],head2[MAX],bin[MAX],stack[MAX],mp[MAX][],father[MAX];
bool mark[MAX]; struct Edge{
int v,id,next;
Edge(){}
Edge(int V,int ID,int NEXT):v(V),id(ID),next(NEXT){}
}edge[MAX*],edge2[MAX*]; void Init(int num){
for(int i=;i<=num;++i)head[i]=head2[i]=-,mark[i]=false;
size=top=;
}
void InsertEdge(int u,int v,int id){
edge[size]=Edge(v,id,head[u]);
head[u]=size++;
}
void InsertEdge2(int u,int v,int id){
edge2[size]=Edge(v,id,head2[u]);
head2[u]=size++;
}
void dfs(int u,int father,int k){
deep[u]=k;
for(int i=head[u];i != -;i=edge[i].next){
int v=edge[i].v;
if(v == father)continue;
dfs(v,u,k+);
}
}
void RMQ(int u,int father){
stack[++top]=u;
int fa=stack[top-];
up[u][]=down[u][]=;
maxw[u][]=minw[u][]=ww[u];
for(int i=;bin[i]<=top;++i){//2^i<=top
fa=stack[top-bin[i-]];
up[u][i]=max(max(up[u][i-],up[fa][i-]),maxw[fa][i-]-minw[u][i-]);
down[u][i]=max(max(down[u][i-],down[fa][i-]),maxw[u][i-]-minw[fa][i-]);
maxw[u][i]=max(maxw[u][i-],maxw[fa][i-]);
minw[u][i]=min(minw[u][i-],minw[fa][i-]);
mp[u][i]=stack[top-bin[i]];
}
for(int i=head[u];i != -;i=edge[i].next){
int v=edge[i].v;
if(v == father)continue;
RMQ(v,u);
}
--top;
}
int findset(int v){
if(v != father[v])father[v]=findset(father[v]);
return father[v];
}
void LCA(int u){
mark[u]=true;
father[u]=u;
for(int i=head2[u];i != -;i=edge2[i].next){
int v=edge2[i].v,id=edge2[i].id;
if(!mark[v])continue;
anc[id]=findset(v);
}
for(int i=head[u];i != -;i=edge[i].next){
int v=edge[i].v;
if(mark[v])continue;
LCA(v);
father[v]=u;
}
}
int search(int x){
int i=;
while(bin[i+]<=x)++i;
return i;
}
int Minw(int u,int anc){
int i=search(deep[u]-deep[anc]+);
if(bin[i] == deep[u]-deep[anc]+)return minw[u][i];
return min(minw[u][i],Minw(mp[u][i],anc));
}
int Maxw(int u,int anc){
int i=search(deep[u]-deep[anc]+);
if(bin[i] == deep[u]-deep[anc]+)return maxw[u][i];
return max(maxw[u][i],Maxw(mp[u][i],anc));
}
int Down(int u,int anc){
int i=search(deep[u]-deep[anc]+);
if(bin[i] == deep[u]-deep[anc]+)return down[u][i];
int downfa=Down(mp[u][i],anc);
downfa=max(downfa,down[u][i]);
int minwfa=Minw(mp[u][i],anc);
return max(downfa,maxw[u][i]-minwfa);
}
int UP(int u,int anc){
int i=search(deep[u]-deep[anc]+);
if(bin[i] == deep[u]-deep[anc]+)return up[u][i];
int upfa=UP(mp[u][i],anc);
upfa=max(upfa,up[u][i]);
int maxwfa=Maxw(mp[u][i],anc);
return max(upfa,maxwfa-minw[u][i]);
}
int main(){
bin[]=;
for(int i=;bin[i-]<MAX;++i)bin[i]=bin[i-]*;
int u,v;
while(~scanf("%d",&n)){
Init(n);
for(int i=;i<=n;++i)scanf("%d",ww+i);
for(int i=;i<n;++i){
scanf("%d%d",&u,&v);
InsertEdge(u,v,i);
InsertEdge(v,u,i);
}
size=;
scanf("%d",&m);
for(int i=;i<m;++i){
scanf("%d%d",uu+i,vv+i);
InsertEdge2(uu[i],vv[i],i);
InsertEdge2(vv[i],uu[i],i);
}
dfs(,-,);
RMQ(,-);
LCA();
for(int i=;i<m;++i){
int upmax=UP(uu[i],anc[i]),downmax=Down(vv[i],anc[i]);
int Minww=Minw(uu[i],anc[i]),Maxww=Maxw(vv[i],anc[i]);
printf("%d\n",max(max(upmax,downmax),Maxww-Minww));
}
}
return ;
}
/*
7
300
11
11
21
10
31
222
1 2
2 3
3 4
4 5
2 6
1 7
1
5 6
*/
04-26 18:50