[SDOI2011]染色(树链剖分)
Description
#include <cstdio>
#include <cstdlib>
#include <vector>
using namespace std;
const int N=1e5+;
struct node
{
int l,r,lc,rc,cl,cr,sum;
bool flag;
}f[N*];
int fa[N],deep[N],son[N],d[N],dfn[N],si[N],top[N],rt,
tot,cnt,n,m,u,v,c,re[N],L,R,a,b;
char s[];
vector <int> link[N];
void dfs1(int u,int f)
{
fa[u]=f;
deep[u]=deep[f]+;
si[u]=;
int size=link[u].size();
for(int i=;i<size;i++)
{
int v=link[u][i];
if(v!=f)
{
dfs1(v,u);
si[u]+=si[v];
if(son[u]== || si[son[u]]<si[v]) son[u]=v;
}
}
}
void dfs2(int u,int f)
{
dfn[u]=++cnt,re[cnt]=u;
if(son[u]!=) top[son[u]]=top[u],dfs2(son[u],u);
int size=link[u].size();
for(int i=;i<size;i++)
{
int v=link[u][i];
if(v!=f && v!=son[u])
top[v]=v,dfs2(v,u);
}
}
void push_up(int g)
{
int lc=f[g].lc,rc=f[g].rc;
f[g].cl=f[lc].cl,f[g].cr=f[rc].cr;
f[g].sum=f[lc].sum+f[rc].sum;
if(f[lc].cr==f[rc].cl) f[g].sum--;
}
void push_down(int g)
{
if(f[g].flag)
{
int lc=f[g].lc,rc=f[g].rc;
f[lc].flag=f[rc].flag=true;
f[lc].cl=f[lc].cr=f[rc].cl=f[rc].cr=f[g].cl;
f[lc].sum=f[rc].sum=;
f[g].flag=false;
}
}
void build(int &g,int l,int r)
{
g=++tot;
f[g].l=l,f[g].r=r;
if(l==r)
{
f[g].flag=true,f[g].sum=;
f[g].cl=f[g].cr=d[re[l]];
return ;
}
int mid=(l+r)>>;
build(f[g].lc,l,mid);
build(f[g].rc,mid+,r);
push_up(g);
}
void add(int g,int l,int r,int c)
{
if(f[g].l>=l && f[g].r<=r)
f[g].flag=true,f[g].cl=f[g].cr=c,f[g].sum=;
else
{
push_down(g);
int mid=(f[g].l+f[g].r)>>;
if(r<=mid) add(f[g].lc,l,r,c);
else if(l>mid) add(f[g].rc,l,r,c);
else add(f[g].lc,l,mid,c),add(f[g].rc,mid+,r,c);
push_up(g);
}
}
void Add(int x,int y,int c)
{
int px=top[x],py=top[y];
while(px!=py)
if(deep[px]>deep[py])
add(rt,dfn[px],dfn[x],c),x=fa[px],px=top[x];
else add(rt,dfn[py],dfn[y],c),y=fa[py],py=top[y];
if(dfn[x]<dfn[y]) add(rt,dfn[x],dfn[y],c);
else add(rt,dfn[y],dfn[x],c);
}
int get(int g,int l,int r)
{
if(f[g].l==L) a=f[g].cl;
if(f[g].r==R) b=f[g].cr;
if(f[g].l>=l && f[g].r<=r)
return f[g].sum;
else
{
push_down(g);
int mid=(f[g].l+f[g].r)>>;
if(r<=mid) return get(f[g].lc,l,r);
else if(l>mid) return get(f[g].rc,l,r);
else
{
int ans=get(f[g].lc,l,mid)+get(f[g].rc,mid+,r);
if(f[f[g].lc].cr==f[f[g].rc].cl) ans--;
return ans;
}
}
}
int Get(int x,int y)
{
int ans=;
int px=top[x],py=top[y];
int l_x=-,l_y=-;
while(px!=py)
if(deep[px]>deep[py])
{
L=dfn[px],R=dfn[x];
ans+=get(rt,dfn[px],dfn[x]);
x=fa[px],px=top[x];
if(b==l_x) ans--;
l_x=a;
}
else
{
L=dfn[py],R=dfn[y];
ans+=get(rt,dfn[py],dfn[y]);
y=fa[py],py=top[y];
if(b==l_y) ans--;
l_y=a;
}
if(dfn[x]>dfn[y]) swap(x,y),swap(l_x,l_y);
L=dfn[x],R=dfn[y];
ans+=get(rt,dfn[x],dfn[y]);
if(a==l_x) ans--;
if(b==l_y) ans--;
return ans;
}
int main()
{
scanf("%d%d",&n,&m);
for(int i=;i<=n;i++) scanf("%d",&d[i]);
for(int i=;i<n;i++)
{
scanf("%d%d",&u,&v);
link[u].push_back(v);
link[v].push_back(u);
}
dfs1(,);
top[]=,dfs2(,);
build(rt,,cnt);
while(m--)
{
scanf("%s%d%d",s,&u,&v);
if(s[]=='C')
{
scanf("%d",&c);
Add(u,v,c);
}
else if(s[]=='Q')
printf("%d\n",Get(u,v));
}
return ;
}
[SDOI2011]染色(树链剖分)的更多相关文章
- BZOJ 2243: [SDOI2011]染色 [树链剖分]
2243: [SDOI2011]染色 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 6651 Solved: 2432[Submit][Status ...
- bzoj-2243 2243: [SDOI2011]染色(树链剖分)
题目链接: 2243: [SDOI2011]染色 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 6267 Solved: 2291 Descript ...
- 【BZOJ2243】[SDOI2011]染色 树链剖分+线段树
[BZOJ2243][SDOI2011]染色 Description 给定一棵有n个节点的无根树和m个操作,操作有2类: 1.将节点a到节点b路径上所有点都染成颜色c: 2.询问节点a到节点b路径上的 ...
- Bzoj 2243: [SDOI2011]染色 树链剖分,LCT,动态树
2243: [SDOI2011]染色 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 5020 Solved: 1872[Submit][Status ...
- bzoj2243[SDOI2011]染色 树链剖分+线段树
2243: [SDOI2011]染色 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 9012 Solved: 3375[Submit][Status ...
- BZOJ 2243: [SDOI2011]染色 树链剖分 倍增lca 线段树
2243: [SDOI2011]染色 Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://www.lydsy.com/JudgeOnline/pr ...
- BZOJ 2243: [SDOI2011]染色 树链剖分+线段树区间合并
2243: [SDOI2011]染色 Description 给定一棵有n个节点的无根树和m个操作,操作有2类: 1.将节点a到节点b路径上所有点都染成颜色c: 2.询问节点a到节点b路径上的颜色段数 ...
- 2243: [SDOI2011]染色(树链剖分+线段树)
2243: [SDOI2011]染色 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 8400 Solved: 3150[Submit][Status ...
- Luogu P2486 [SDOI2011]染色(树链剖分+线段树合并)
Luogu P2486 [SDOI2011]染色 题面 题目描述 输入输出格式 输入格式: 输出格式: 对于每个询问操作,输出一行答案. 输入输出样例 输入样例: 6 5 2 2 1 2 1 1 1 ...
- [bzoj 2243]: [SDOI2011]染色 [树链剖分][线段树]
Description 给定一棵有n个节点的无根树和m个操作,操作有2类: 1.将节点a到节点b路径上所有点都染成颜色c: 2.询问节点a到节点b路径上的颜色段数量(连续相同颜色被认为是同一段),如“ ...
随机推荐
- Netty小结
前言 在实际开发中,netty的开发使用相对较小,why?在企业中涉及网络编程的部分比重较小,在这大环境内,企业会优先使用简单的http,udp等基础的通讯协议工具,如果不能满足需求,会考虑基于rpc ...
- E420笔记本升级固态硬盘
后壳比较好拆 机械硬盘盒 这里 可费了好大劲 才 拧开这 4个螺丝 光驱支架买的这个38-5rmb,京东自提 以前不知道的一件事: 原来的光驱挡板要自己拆下来换到新买的光驱支架上面 ...
- HashMap、lru、散列表
HashMap HashMap的数据结构:HashMap实际上是一个数组和链表("链表散列")的数据结构.底层就是一个数组结构,数组中的每一项又是一个链表. hashCode是一个 ...
- MySQL基础之事务编程学习笔记
MySQL基础之事务编程学习笔记 在学习<MySQL技术内幕:SQL编程>一书,并做了笔记.本博客内容是自己学了<MySQL技术内幕:SQL编程>事务编程一章之后,根据自己的理 ...
- Spring||Interview
1.依赖注入(DI)(IOC) 对象本身不负责对象的创建和维护,将控制权转交给外部的容器实现,降低程序的耦合度,只提供java方法让容器决定依赖关系,依赖关系的对象通过JavaBean属性或者构造函数 ...
- 分析CPU使用率不断增加的原因
工程中发现引起的问题: 结合别的朋友的意见,我的优化思路是: 1.排查是否内存泄漏 经过反复查询代码,未发现有内存泄漏(可以自己百度搜索C#内存泄漏的原因).可以通过任务管理器分析是否有内存泄漏,打开 ...
- windows 服务的安装、启动、状态查询、停止操作c++实现
具体的自己看看代码 粘贴复制即可使用 卸载也很简单自己查看MSDN 加上就是 #ifndef __SERVICEMANAGE_H__ #define __SERVICEMANAGE_H__ #incl ...
- 洛谷训练新手村之“BOSS战-入门综合练习1”题解
P1478 陶陶摘苹果(升级版) 题目链接:https://www.luogu.com.cn/problem/P1478 题目大意:陶陶有s点体力值,每个苹果消耗体力值,问s体力值最多能摘多少苹果. ...
- 推荐中的多任务学习-YouTube视频推荐
本文将介绍Google发表在RecSys'19 的论文<Recommending What Video to Watch Next: A Multitask Ranking System> ...
- owa部署
新建一台win server 2012(注意如果是2008要补丁) 配置静态ip DNS指向ad域的ip 测试: ping 下ad域的域名,是通的继续 把本机加入到ad域 重启下 用admin登陆: ...