[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路径上的颜色段数量(连续相同颜色被认为是同一段),如“ ...
随机推荐
- 安卓中运行报错Error:Execution failed for task ':app:transformClassesWithDexForDebug'解决
在androidstuio中运行我的未完项目,报错: Error:Execution failed for task ':app:transformClassesWithDexForDebug'.&g ...
- 小白学 Python 爬虫(31):自己构建一个简单的代理池
人生苦短,我用 Python 前文传送门: 小白学 Python 爬虫(1):开篇 小白学 Python 爬虫(2):前置准备(一)基本类库的安装 小白学 Python 爬虫(3):前置准备(二)Li ...
- 【题解】有标号的DAG计数2
[HZOI 2015] 有标号的DAG计数 II \(I\)中DP只有一个数组, \[ dp_i=\sum{i\choose j}2^{j(i-j)}dp_{i-j}(-1)^{j+1} \] 不会. ...
- 洛谷P3413 SAC#1 - 萌数 题解 数位DP
题目链接:https://www.luogu.com.cn/problem/P3413 题目大意: 定义萌数指:满足"存在长度至少为2的回文子串"的数. 求区间 \([L,R]\) ...
- 「2015南阳CCPC D」金砖 解题报告
金砖 Problem 有一个长度为L的板凳,可以放一排金砖,金砖不能重叠.特别的,摆放的金砖可以超出板凳,前提是必须保证该金砖不会掉下去,即该金砖的重心必须在板凳上. 每块金砖都一个长度和价值,且金砖 ...
- [ASP.NET Core 3框架揭秘] Options[2]: 配置选项的正确使用方式[下篇]
四.直接初始化Options对象 前面演示的几个实例具有一个共同的特征,即都采用配置系统来提供绑定Options对象的原始数据,实际上,Options框架具有一个完全独立的模型,可以称为Options ...
- ENS 域名注册表智能合约(ENSRegistry.sol)解析
ENS 注册表合约是 ENS 系统中的核心合约,了解这个合约可以敲开我们理解 ENS 域名系统的大门. 打开下面的折叠区域可以查看用 Solidity 语言编写的详细代码.当前部署在以太坊中的 ENS ...
- RabbitMQ入门之Hello World
RabbitMQ简介 在介绍RabbitMQ之前,我们需要了解一些最基础的概念,相信使用过或者听说过RabbitMQ的人都不会陌生,但笔者还是不厌其烦地在这里讲述,因为笔者的理念是self con ...
- 关于java php go 中AES加解密秘钥长度问题
今天心血来朝,想用go把php中的一个小功能重写一下,但在解密aes加密的数据时碰到了个坑! php的mcrypt拓展(貌似php7.1版本以上不支持了)提供了aes的加解密: 而且php aes 的 ...
- hive 动态分区
非常重要的动态分区属性: hive.exec.dynamic.partition 是否启动动态分区.false(不开启) true(开启)默认是 false hive.exec.dynamic.pa ...