题目链接:https://vjudge.net/contest/279350#problem/A

题目大意:n个点,m条边,然后q次询问,因为在树上,两个点能确定一条直线,我们可以对这条直线上的所有值进行加减操作,也可以单点询问。

各个数组的作用:sto是刚开始的输入数据,head是前向星,dfsnum指的是dfs序,depth指的是每个点的深度son指的是每个节点的重儿子,father指的是每个点的父节点,Size指的是以当前点为根节点的树,ord指的是遍历顺序,cost指的是编号之后的每个点,top指的是当前的这条重链的最顶端的那个点,剩下的就是线段树的数组了。

注意点:我们通过两个dfs来给这些数组赋值,通过第一个dfs,我们可以把depth和father,size,son求出来,剩下的ord和top通过第二个dfs求出来,为什么使用两个dfs?我的理解就是,第一个dfs和第二个dfs的遍历条件并不相同,第一个dfs就是能走就走,第二个dfs是在按照已经分好链的前提下进行走的,也就是说这里的ord数组并不能在第一个dfs中实现,只能在第二个数组中实现。(后续有新的理解会继续补充)。

AC代码:

  1 #include<iostream>
  2 #include<cmath>
  3 #include<stack>
  4 #include<queue>
  5 #include<stdio.h>
  6 #include<string>
  7 #include<cstring>
  8 #include<algorithm>
  9 using namespace std;
 10 # define inf 0x3f3f3f3f
 11 # define ll long long
 12 # define lson l,m,rt<<1
 13 # define rson m+1,r,rt<<1|1
 14 const int maxn = 5e4+100;
 15 int sto[maxn],head[maxn],edgnum,dfsnum,depth[maxn];
 16 int son[maxn],father[maxn],Size[maxn],ord[maxn],cost[maxn],top[maxn];
 17 int tree[maxn<<2],lazy[maxn<<2];
 18 struct node
 19 {
 20     int to;
 21     int nex;
 22 } edge[maxn<<2];
 23 void addedge(int fr,int to)
 24 {
 25     edge[edgnum].nex=head[fr];
 26     edge[edgnum].to=to;
 27     head[fr]=edgnum++;
 28 }
 29 void dfs1(int fr,int rt,int dep)
 30 {
 31     father[fr]=rt;
 32     Size[fr]=1;
 33     son[fr]=-1;
 34     depth[fr]=dep;
 35     for(int i=head[fr]; i!=-1; i=edge[i].nex)
 36     {
 37         int to=edge[i].to;
 38         if(to==rt)
 39             continue;
 40         dfs1(to,fr,dep+1);
 41         Size[fr]+=Size[to];
 42         if(son[to]==-1||(Size[son[fr]]<Size[to]))
 43         {
 44             son[fr]=to;
 45         }
 46     }
 47 }
 48 void dfs2(int fr,int rt)
 49 {
 50     ord[fr]=++dfsnum;
 51     cost[ord[fr]]=sto[fr];
 52     top[fr]=rt;
 53     if(son[fr]!=-1)
 54         dfs2(son[fr],rt);
 55     for(int i=head[fr]; i!=-1; i=edge[i].nex)
 56     {
 57         int u=edge[i].to;
 58         if(son[fr]!=u&&father[fr]!=u)
 59         {
 60             dfs2(u,u);
 61         }
 62     }
 63 }
 64 void init()
 65 {
 66     dfsnum=0;
 67     dfs1(1,-1,1);
 68     dfs2(1,1);
 69 }
 70 void up(int rt)
 71 {
 72     tree[rt]=tree[rt<<1]+tree[rt<<1|1];
 73 }
 74 void down(int len,int rt)
 75 {
 76     if(lazy[rt])
 77     {
 78         lazy[rt<<1]+=lazy[rt];
 79         lazy[rt<<1|1]+=lazy[rt];
 80         tree[rt<<1]+=(len-len/2)*lazy[rt];
 81         tree[rt<<1|1]+=(len/2)*lazy[rt];
 82         lazy[rt]=0;
 83     }
 84 }
 85 void buildtree(int l,int r,int rt)
 86 {
 87     lazy[rt]=0;
 88     tree[rt]=0;
 89     if(l==r)
 90     {
 91         tree[rt]=cost[l];
 92         return ;
 93     }
 94     int m=(l+r)>>1;
 95     buildtree(lson);
 96     buildtree(rson);
 97     up(rt);
 98 }
 99 void update(int l,int r,int rt,int L,int R,int p)
100 {
101     if(L<=l&&R>=r)
102     {
103         tree[rt]+=p*(r-l+1);
104         lazy[rt]+=p;
105         return ;
106     }
107     down(r-l+1,rt);
108     int m=(l+r)>>1;
109     if(L<=m)
110         update(lson,L,R,p);
111     if(R>m)
112         update(rson,L,R,p);
113     up(rt);
114 }
115 void Update(int n,int x,int y,int p)
116 {
117     int tx=top[x],ty=top[y];
118     while(tx!=ty)
119     {
120         if(depth[tx]<depth[ty])
121         {
122             swap(tx,ty);
123             swap(x,y);
124         }
125         update(1,n,1,ord[tx],ord[x],p);
126         x=father[tx],tx=top[x];
127     }
128     if(depth[x]<depth[y])
129     {
130         swap(x,y);
131     }
132     update(1,n,1,ord[y],ord[x],p);
133 }
134 int query(int l,int r,int rt,int pos)
135 {
136     if(l==r)
137     {
138         return tree[rt];
139     }
140     down(r-l+1,rt);
141     int ans=0;
142     int m=(l+r)>>1;
143     if(pos<=m)
144         ans+=query(lson,pos);
145     if(pos>m)
146         ans+=query(rson,pos);
147     return ans;
148     up(rt);
149 }
150 int main()
151 {
152     int n,m,q;
153     while(~scanf("%d %d %d",&n,&m,&q))
154     {
155         edgnum=0;
156         for(int i=1; i<=n; i++)
157         {
158             scanf("%d",&sto[i]);
159             head[i]=-1;
160         }
161         int t1,t2;
162         for(int i=1; i<=m; i++)
163         {
164             scanf("%d %d",&t1,&t2);
165             addedge(t1,t2);
166             addedge(t2,t1);
167         }
168         init();
169         char str[10];
170         buildtree(1,n,1);
171         int t3;
172         while(q--)
173         {
174             scanf("%s",str);
175             if(str[0]=='I')
176             {
177                 scanf("%d %d %d",&t1,&t2,&t3);
178                 Update(n,t1,t2,t3);
179             }
180             else if(str[0]=='Q')
181             {
182                 scanf("%d",&t1);
183                 int ans=query(1,n,1,ord[t1]);
184                 printf("%d\n",ans);
185             }
186             else if(str[0]=='D')
187             {
188                 scanf("%d %d %d",&t1,&t2,&t3);
189                 Update(n,t1,t2,-t3);
190             }
191         }
192     }
193     return 0;
194 }