2243. [SDOI2011]染色【树链剖分】

Description

给定一棵有n个节点的无根树和m个操作,操作有2类:
1、将节点a到节点b路径上所有点都染成颜色c;
2、询问节点a到节点b路径上的颜色段数量(连续相同颜色被认为是同一段),
如“112221”由3段组成:“11”、“222”和“1”。
请你写一个程序依次完成这m个操作。

Input

第一行包含2个整数n和m,分别表示节点数和操作数;
第二行包含n个正整数表示n个节点的初始颜色
下面 行每行包含两个整数x和y,表示x和y之间有一条无向边。
下面 行每行描述一个操作:
“C a b c”表示这是一个染色操作,把节点a到节点b路径上所有点(包括a和b)都染成颜色c;
“Q a b”表示这是一个询问操作,询问节点a到节点b(包括a和b)路径上的颜色段数量。

Output

对于每个询问操作,输出一行答案。

Sample Input

6 5
2 2 1 2 1 1
1 2
1 3
2 4
2 5
2 6
Q 3 5
C 2 1 1
Q 3 5
C 5 1 2
Q 3 5

Sample Output

3
1
2

HINT

数N<=10^5,操作数M<=10^5,所有的颜色C为整数且在[0, 10^9]之间。

 

一眼:链剖
两眼:链剖+两个区间相邻处特判
和学长说了一句:感觉不是很难写(flag)
然后我处理的时候是直接当区间修改时用for循环将线段树区间更新
……正确性当然可以保证啦……出乎意料的是并没有T飞……就T了一个点
去隔壁请教了学长被告知要维护线段树的两端颜色……
于是开始了漫长的调试……
其实这个题只要想到了维护两端的值就很好搞了啊……
哪怕像我想不到维护两端也有95可拿对不对【滑稽】

 

  1 #include<iostream>
  2 #include<cstdio>
  3 #include<cstring>
  4 #include<cstdlib>
  5 #define MAX (150000+5)
  6 using namespace std;
  7 struct node
  8 {
  9     int down;//下传标记
 10     int sum;//该区间有几种颜色
 11     int l,r;//该区间左右端点的颜色
 12 } Segt[MAX*4];
 13 struct node1
 14 {
 15     int to;
 16     int next;
 17 } edge[MAX*2];
 18 int Father[MAX],Depth[MAX];
 19 int Sum[MAX],Son[MAX],Top[MAX];
 20 int TREE[MAX],T_NUM[MAX];
 21 int num_edge,head[MAX],n,cnt;
 22 int a[MAX];
 23 
 24 inline int get()//神TM快读,维护左右颜色前卡常用的但没卡过去
 25 {
 26     char c;
 27     int x=0,f=1;
 28     c=getchar();
 29     while (c<'0'||c>'9')
 30     {
 31         if (c=='-') f=-1;
 32         c=getchar();
 33     }
 34     while (c>='0'&&c<='9')
 35     {
 36         x=x*10+c-'0';
 37         c=getchar();
 38     }
 39     return x*f;
 40 }
 41 
 42 void add(int u,int v)
 43 {
 44     edge[++num_edge].to=v;
 45     edge[num_edge].next=head[u];
 46     head[u]=num_edge;
 47 }
 48 
 49 void Build(int node,int l,int r)
 50 {
 51     if (l==r)
 52     {
 53         Segt[node].sum=1;
 54         Segt[node].l=Segt[node].r=TREE[l];
 55     }
 56     else
 57     {
 58         int mid=(l+r)/2;
 59         Build(node*2,l,mid);
 60         Build(node*2+1,mid+1,r);//这里维护好麻烦啊……早知道写个pushup了
 61         Segt[node].l=Segt[node*2].l;
 62         Segt[node].r=Segt[node*2+1].r;
 63         Segt[node].sum=Segt[node*2].sum+Segt[node*2+1].sum-(Segt[node*2].r==Segt[node*2+1].l);
 64     }
 65 }
 66 
 67 void Pushdown(int node)
 68 {
 69     if (Segt[node].down!=0)
 70     {
 71         Segt[node*2].sum=1;//因为修改成了一个数所以肯定只有一种颜色
 72         Segt[node*2+1].sum=1;
 73 
 74         Segt[node*2].down=Segt[node].down;//down存下传的颜色编号
 75         Segt[node*2+1].down=Segt[node].down;
 76 
 77         Segt[node*2].l=Segt[node*2].r=Segt[node].down;
 78         Segt[node*2+1].l=Segt[node*2+1].r=Segt[node].down;
 79 
 80         Segt[node].down=0;
 81     }
 82 }
 83 
 84 int Query(int node,int l,int r,int l1,int r1)
 85 {
 86     if (r<l1 || l>r1)
 87         return 0;
 88     if (l1<=l && r<=r1)
 89         return Segt[node].sum;
 90     Pushdown(node);
 91     int mid=(l+r)/2;
 92     int x=Query(node*2,l,mid,l1,r1);
 93     int y=Query(node*2+1,mid+1,r,l1,r1);
 94     if (x!=0 && y!=0)
 95         return x+y-(Segt[node*2].r==Segt[node*2+1].l);
 96     return max(x,y);
 97 }
 98 
 99 void Update(int node,int l,int r,int l1,int r1,int k)//区间更新模板
100 {
101     if (r<l1 || l>r1)
102         return;
103     if (l1<=l && r<=r1)
104     {
105         Segt[node].down=k;
106         Segt[node].sum=1;
107         Segt[node].l=Segt[node].r=k;
108         return;
109     }
110     Pushdown(node);
111     int mid=(l+r)/2;
112     Update(node*2,l,mid,l1,r1,k);
113     Update(node*2+1,mid+1,r,l1,r1,k);//再一次后悔没写pushup
114     Segt[node].l=Segt[node*2].l;
115     Segt[node].r=Segt[node*2+1].r;
116     Segt[node].sum=Segt[node*2].sum+Segt[node*2+1].sum-(Segt[node*2].r==Segt[node*2+1].l);
117 }
118 
119 int Ask(int node,int l,int r,int x)//查询某个点的颜色
120 {
121     if (l==r)
122         return Segt[node].l;
123     else
124     {
125         Pushdown(node);
126         int mid=(l+r)/2;
127         if (x<=mid)    return Ask(node*2,l,mid,x);
128         else return Ask(node*2+1,mid+1,r,x);
129     }
130 }
131 
132 int Get(int x,int y)
133 {
134     int fx=Top[x],fy=Top[y];
135     int Ans=0;
136     while (fx!=fy)
137     {
138         if (Depth[fx]<Depth[fy])
139             swap(fx,fy),swap(x,y);
140         int re=Ask(1,1,n,T_NUM[fx]);
141         int fun=Ask(1,1,n,T_NUM[Father[fx]]);
142         Ans+=Query(1,1,n,T_NUM[fx],T_NUM[x])-(re==fun);
143         x=Father[fx],fx=Top[x];
144     }
145     if (Depth[x]<Depth[y])
146         swap(x,y);
147     return Ans+=Query(1,1,n,T_NUM[y],T_NUM[x]);
148 }
149 
150 void Change(int x,int y,int k)
151 {
152     int fx=Top[x],fy=Top[y];
153     while (fx!=fy)
154     {
155         if (Depth[fx]<Depth[fy])
156             swap(fx,fy),swap(x,y);
157         Update(1,1,n,T_NUM[fx],T_NUM[x],k);
158         x=Father[fx],fx=Top[x];
159     }
160     if (Depth[x]<Depth[y])
161         swap(x,y);
162     Update(1,1,n,T_NUM[y],T_NUM[x],k);
163 }
164 
165 void Dfs1(int x)
166 {
167     Depth[x]=Depth[Father[x]]+1;
168     Sum[x]=1;
169     for (int i=head[x]; i!=0; i=edge[i].next)
170         if (edge[i].to!=Father[x])
171         {
172             Father[edge[i].to]=x;
173             Dfs1(edge[i].to);
174             Sum[x]+=Sum[edge[i].to];
175             if (Son[x]==0 || Sum[Son[x]]<Sum[edge[i].to])
176                 Son[x]=edge[i].to;
177         }
178 }
179 
180 void Dfs2(int x,int pre)
181 {
182     TREE[++cnt]=a[x];
183     T_NUM[x]=cnt;
184     Top[x]=pre;
185     if (Son[x]!=0)
186         Dfs2(Son[x],pre);
187     for (int i=head[x]; i!=0; i=edge[i].next)
188         if (edge[i].to!=Son[x] && edge[i].to!=Father[x])
189             Dfs2(edge[i].to,edge[i].to);
190 }
191 
192 int main()
193 {
194     char ch;
195     int x,y,k,m,u,v;
196     n=get();
197     m=get();
198     for (int i=1; i<=n; ++i)
199         a[i]=get();
200     for (int i=1; i<=n-1; ++i)
201     {
202         u=get();
203         v=get();
204         add(u,v);
205         add(v,u);
206     }
207     Dfs1(1);
208     Father[1]=1;
209     Dfs2(1,1);
210     Build(1,1,n);
211     for (int i=1; i<=m; ++i)
212     {
213         ch=getchar();
214         while (ch!='C'&&ch!='Q') ch=getchar();
215         if (ch=='Q')
216             x=get(),y=get(),printf("%d\n",Get(x,y));
217         else
218             x=get(),y=get(),k=get(),Change(x,y,k);
219     }
220 }
posted @ 2018-03-30 20:48  Refun  阅读(143)  评论(0编辑  收藏  举报