P2486 [SDOI2011]染色 区间合并+树链剖分(加深对线段树的理解)
#include<bits/stdc++.h>
using namespace std;
const int M=3e5+;
struct node{
int l,r,cnt,lazy;
node(int l1=,int r1=,int cnt1=,int lazy1=):l(l1),r(r1),cnt(cnt1),lazy(lazy1){}
}tree[M<<];
int fa[M],sz[M],deep[M],dfn[M],son[M],to[M],a[M],top[M],cnt,n;
char s[];
vector<int>g[M];
void dfs1(int u,int from){
fa[u]=from;
sz[u]=;
deep[u]=deep[from]+;
for(int i=;i<g[u].size();i++){ int v=g[u][i];
if(v!=from){
dfs1(v,u);
sz[u]+=sz[v];
if(sz[v]>sz[son[u]])
son[u]=v;
} }
}
void dfs2(int u,int t){
top[u]=t;
dfn[u]=++cnt;
to[cnt]=u;
if(!son[u])
return ;
dfs2(son[u],t);
for(int i=;i<g[u].size();i++){
int v=g[u][i];
if(v!=fa[u]&&v!=son[u])
dfs2(v,v);
}
}
void up(int root){
tree[root].cnt=tree[root<<].cnt+tree[root<<|].cnt;
if(tree[root<<].r==tree[root<<|].l)
tree[root].cnt--;
tree[root].l=tree[root<<].l;
tree[root].r=tree[root<<|].r;
}
void build(int root,int l,int r){
tree[root].lazy=;
if(l==r){
tree[root].l=tree[root].r=a[to[l]];
tree[root].cnt=;
return ;
}
int midd=(l+r)>>;
build(root<<,l,midd);
build(root<<|,midd+,r);
up(root);
}
void pushdown(int root){
tree[root<<]=tree[root<<|]=node(tree[root].l,tree[root].r,,tree[root].lazy);
tree[root].lazy=;
}
void update(int L,int R,int x,int root,int l,int r){
if(L<=l&&r<=R){
tree[root]=node(x,x,,x);
return ;
}
if(tree[root].lazy)
pushdown(root);
int midd=(l+r)>>;
if(L<=midd)
update(L,R,x,root<<,l,midd);
if(R>midd)
update(L,R,x,root<<|,midd+,r);
up(root);
}
void add(int u,int v ,int w){
int fu=top[u],fv=top[v];
while(fu!=fv){
if(deep[fu]>=deep[fv])
update(dfn[fu],dfn[u],w,,,n),u=fa[fu],fu=top[u];
else
update(dfn[fv],dfn[v],w,,,n),v=fa[fv],fv=top[v];
}
if(dfn[u]<=dfn[v])
update(dfn[u],dfn[v],w,,,n);
else
update(dfn[v],dfn[u],w,,,n);
}
node meger(node a,node b){
if(!a.cnt)
return b;
if(!b.cnt)
return a;
node ans=node(,,,);
ans.cnt=a.cnt+b.cnt;
if(a.r==b.l)
ans.cnt--;
ans.l=a.l;
ans.r=b.r;
return ans;
}
node query(int L,int R,int root,int l,int r){
if(L<=l&&r<=R){
return tree[root];
}
if(tree[root].lazy)
pushdown(root);
int midd=(l+r)>>;
node ans;
if(L<=midd)
ans=query(L,R,root<<,l,midd);
if(R>midd)
ans=meger(ans,query(L,R,root<<|,midd+,r));
up(root);
return ans;
}
int solve(int u,int v){
node l,r;
int fv=top[v],fu=top[u];
while(fv!=fu){
if(deep[fu]>=deep[fv])
l=meger(query(dfn[fu],dfn[u],,,n),l),u=fa[fu],fu=top[u];
else
r=meger(query(dfn[fv],dfn[v],,,n),r),v=fa[fv],fv=top[v];
}
if(dfn[u]<=dfn[v])
r=meger(query(dfn[u],dfn[v],,,n),r);
else
l=meger(query(dfn[v],dfn[u],,,n),l);
swap(l.l,l.r);
l=meger(l,r);
return l.cnt;
}
int main(){
int m;
scanf("%d%d",&n,&m);
for(int i=;i<=n;i++)
scanf("%d",&a[i]);
for(int i=;i<n;i++){
int u,v;
scanf("%d%d",&u,&v);
g[u].push_back(v);
g[v].push_back(u);
}//cout<<"!!"<<endl;
dfs1(,);
dfs2(,); build(,,n);
while(m--){
int u,v,w;
scanf("%s",s);
if(s[]=='Q'){
scanf("%d%d",&u,&v);
printf("%d\n",solve(u,v));
}
else{
scanf("%d%d%d",&u,&v,&w);
add(u,v,w);
}
}
return ;
}
P2486 [SDOI2011]染色 区间合并+树链剖分(加深对线段树的理解)的更多相关文章
- BZOJ 2243: [SDOI2011]染色 树链剖分 倍增lca 线段树
2243: [SDOI2011]染色 Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://www.lydsy.com/JudgeOnline/pr ...
- BZOJ4012[HNOI2015]开店——树链剖分+可持久化线段树/动态点分治+vector
题目描述 风见幽香有一个好朋友叫八云紫,她们经常一起看星星看月亮从诗词歌赋谈到 人生哲学.最近她们灵机一动,打算在幻想乡开一家小店来做生意赚点钱.这样的 想法当然非常好啦,但是她们也发现她们面临着一个 ...
- HDU 5029 Relief grain 树链剖分打标记 线段树区间最大值
Relief grain Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://acm.hdu.edu.cn/showproblem.php?pid= ...
- LOJ2269 [SDOI2017] 切树游戏 【FWT】【动态DP】【树链剖分】【线段树】
题目分析: 好题.本来是一道好的非套路题,但是不凑巧的是当年有一位国家集训队员正好介绍了这个算法. 首先考虑静态的情况.这个的DP方程非常容易写出来. 接着可以注意到对于异或结果的计数可以看成一个FW ...
- [GDOI2016] 疯狂动物园 [树链剖分+可持久化线段树]
题面 太长了,而且解释的不清楚,我来给个简化版的题意: 给定一棵$n$个点的数,每个点有点权,你需要实现以下$m$个操作 操作1,把$x$到$y$的路径上的所有点的权值都加上$delta$,并且更新一 ...
- HYSBZ 4034 【树链剖分】+【线段树 】
<题目链接> 题目大意: 有一棵点数为 N 的树,以点 1 为根,且树点有权值.然后有 M 个 操作,分为三种: 操作 1 :把某个节点 x 的点权增加 a . 操作 2 :把某个节点 x ...
- HDU 3966 Aragorn's Story(模板题)【树链剖分】+【线段树】
<题目链接> 题目大意: 给定一颗带点权的树,进行两种操作,一是给定树上一段路径,对其上每个点的点权增加或者减少一个数,二是对某个编号点的点权进行查询. 解题分析: 树链剖分的模板题,还不 ...
- 焦作网络赛E-JiuYuanWantstoEat【树链剖分】【线段树】
You ye Jiu yuan is the daughter of the Great GOD Emancipator. And when she becomes an adult, she wil ...
- 【树链剖分】【线段树】bzoj3626 [LNOI2014]LCA
引用题解: http://blog.csdn.net/popoqqq/article/details/38823457 题目大意: 给出一个n个节点的有根树(编号为0到n-1,根节点为0).一个点的深 ...
- 【Codeforces】【网络流】【树链剖分】【线段树】ALT (CodeForces - 786E)
题意 现在有m个人,每一个人都特别喜欢狗.另外还有一棵n个节点的树. 现在每个人都想要从树上的某个节点走到另外一个节点,且满足要么这个人自带一条狗m,要么他经过的所有边h上都有一条狗. 2<=n ...
随机推荐
- POJ - 3665 iCow(模拟)
题意:有N首歌曲,播放的顺序按照一定的规则,输出前T首被播放的歌的编号.规则如下: 1.每首歌有一个初始的等级r,每次都会播放当前所有歌曲中r最大的那首歌(若r最大的有多首,则播放编号最小的那首歌). ...
- SpringCloud学习之Stream消息驱动【默认通道】(十)
在实际开发过程中,服务与服务之间通信经常会使用到消息中间件,而以往使用了中间件比如RabbitMQ,那么该中间件和系统的耦合性就会非常高,如果我们要替换为Kafka那么变动会比较大,这时我们可以使用S ...
- MBR&/BOOT&GRUB
能正常工作的grub应该包 括一下文件:stage1.stage2.*stage1_5.menu.lst. 其中stage1要被安装(也就是写入)某个硬盘的主引导记录,或者某个活动分区(这个分区要 ...
- ZOJ-1234 UVA-10271 DP
最近觉得动态规划真的很练脑子,对建模以及思维方法有很大帮助,线段树被卡到有点起不来的感觉 最近仔细思考了一下动态规划的思想,无非是由局部最优解得到全局最优解,由此类推,发现,像最短路和最小生成树其实都 ...
- python:创建文件夹:写入文本1:读取txt:读取Excel文件遍历文件夹:
https://blog.csdn.net/u011956147/article/details/80369731 创建文件夹: import osimport shutil def buildfil ...
- MySQL--SQL分类
SQL语句主要可以划分为以下3个类别: DDL(Data Definition Languages)语句:数据定义语言,这些语句定义了不同的数据段.数据库.表.列.索引等数据库对象. 常用的语句关键字 ...
- 内存管理之堆heap
1.什么是堆? 堆(heap)是一种内存管理方式.内存管理对操作系统来说是一件非常复杂的事情,因为首先内存容量很大, 其次就是内存需求在时间和大小块上没有规律(操作系统上运行着几十甚至几百个进程,这些 ...
- IDEA忽略文件,防止git提交不想提交的文件
IDEA忽略文件,防止git提交不想提交的文件 方法一(只对没有add到仓库的文件有效): 方法二(只对没有add到仓库的文件有效): 在IDEA中安装.ignore插件.创建好了之后: 安装.git ...
- share团队冲刺6
团队冲刺第六天 昨天:进行各种原件的自定义样式,进行界面布局 登陆界面: 今天:进行后台的代码编写,实现各种按钮的功能 问题:在不同的型号手机上,界面会发生不兼容的问题.
- 【C#并发】00概述
摘自<C#并发编程经典实例>[美]Stephen Cleary 并发:同时做多件事情.终端用户利用并发功能,在输入数据库的同时相应用户输入.服务器应用并发,在处理第一个请求的同时响应第二个 ...