Description

给您一颗树,每个节点有个初始值。
现在支持以下两种操作:
1. C i x(0<=x<2^31) 表示将i节点的值改为x。
2. Q i j x(0<=x<2^31) 表示询问i节点到j节点的路径上有多少个值为x的节点。

Input

第一行有两个整数N,Q(1 ≤N≤ 100,000;1 ≤Q≤ 200,000),分别表示节点个数和操作个数。
下面一行N个整数,表示初始时每个节点的初始值。
接下来N-1行,每行两个整数x,y,表示x节点与y节点之间有边直接相连(描述一颗树)。
接下来Q行,每行表示一个操作,操作的描述已经在题目描述中给出。

Output

对于每个Q输出单独一行表示所求的答案。

Sample Input

5 6
10 20 30 40 50
1 2
1 3
3 4
3 5
Q 2 3 40
C 1 40
Q 2 3 40
Q 4 5 30
C 3 10
Q 4 5 30

Sample Output

0
1
1
0

题意:给定一个数,顶点有颜色。 Q次操作,或修改单点颜色,或查询路径颜色为x的个数。

思路:由于是路径,想到树剖+线段树,但是颜色个数无法合并,因此不行。  换个思路,用差分来求,然后把每个颜色弄个线段树,然后这个颜色的线段树单点代表的是这个点到根有多少这个颜色。    如果x点加了一个颜色为y的,那么以y这棵树,就再某个范围加1,这个范围是x的子树的DFS序范围;同理,减少则为-1 。

那么查询u到v路径为x颜色的个数,就在x这棵树上求sum[u]+sum[v]-sum[LCA]-sum[fa[LCA]];

自己写了一遍,感觉动态开点好像也没那么难。 但是为什么我的那么慢啊。

(颜色需要离散化。

#include<bits/stdc++.h>
#define rep(i,a,b) for(int i=a;i<=b;i++)
using namespace std;
const int maxn=;
struct in{
int lson,rson,lazy;
in(){lson=rson=lazy=;}
}s[maxn*];
struct qqq{
char opt[]; int u,v,x;
}q[maxn<<];
int dep[maxn],a[maxn],rt[maxn*],fa[maxn][],in[maxn],ou[maxn],Log[maxn];
int Laxt[maxn],Next[maxn<<],To[maxn<<],cnt,times,b[maxn*],tot;
void add(int u,int v){
Next[++cnt]=Laxt[u]; Laxt[u]=cnt; To[cnt]=v;
}
int LCA(int u,int v){
if(dep[u]<dep[v]) swap(u,v);
for(int i=Log[dep[u]-dep[v]];i>=;i--)
if(dep[fa[u][i]]>=dep[v]) u=fa[u][i];
if(u==v) return u;
for(int i=;i>=;i--)
if(fa[u][i]!=fa[v][i]) u=fa[u][i],v=fa[v][i];
return fa[u][];
}
void dfs(int u,int f){
in[u]=++times;dep[u]=dep[f]+;
for(int i=Laxt[u];i;i=Next[i])
if(To[i]!=f) dfs(To[i],u);
ou[u]=times; fa[u][]=f;
}
void pushdown(int Now){
if(s[Now].lazy!=){
if(!s[Now].lson) s[Now].lson=++cnt;
if(!s[Now].rson) s[Now].rson=++cnt;
s[s[Now].lson].lazy+=s[Now].lazy;
s[s[Now].rson].lazy+=s[Now].lazy;
s[Now].lazy=;
}
}
void addnum(int &Now,int L,int R,int l,int r,int add){
if(!Now) Now=++cnt;
if(l<=L&&r>=R){ s[Now].lazy+=add; return ; }
int Mid=(L+R)>>; pushdown(Now);
if(l<=Mid) addnum(s[Now].lson,L,Mid,l,r,add);
if(r>Mid) addnum(s[Now].rson,Mid+,R,l,r,add);
}
int query(int Now,int L,int R,int pos){
if(!Now) return ;
if(L==R) return s[Now].lazy;
int Mid=(L+R)>>; pushdown(Now);
if(pos<=Mid) return query(s[Now].lson,L,Mid,pos);
return query(s[Now].rson,Mid+,R,pos);
}
int main()
{
int N,Q,u,v,x;
scanf("%d%d",&N,&Q); rep(i,,N) Log[i]=Log[i>>]+;
rep(i,,N) scanf("%d",&a[i]),b[++tot]=a[i];
rep(i,,N-){
scanf("%d%d",&u,&v);
add(u,v); add(v,u);
}
dfs(,); cnt=;
rep(j,,)
rep(i,,N){
fa[i][j]=fa[fa[i][j-]][j-];
}
rep(i,,Q){
scanf("%s%d%d",q[i].opt,&q[i].u,&q[i].v);
if(q[i].opt[]=='Q') scanf("%d",&q[i].x),b[++tot]=q[i].x;
else b[++tot]=q[i].v;
}
sort(b+,b+tot+); tot=unique(b+,b+tot+)-(b+);
rep(i,,N) a[i]=lower_bound(b+,b+tot+,a[i])-b;
rep(i,,Q) {
if(q[i].opt[]=='Q') q[i].x=lower_bound(b+,b+tot+,q[i].x)-b;
else q[i].v=lower_bound(b+,b+tot+,q[i].v)-b;
}
rep(i,,N) addnum(rt[a[i]],,N,in[i],ou[i],);
rep(i,,Q){
u=q[i].u; v=q[i].v;
if(q[i].opt[]=='C'){
addnum(rt[a[u]],,N,in[u],ou[u],-); a[u]=v;
addnum(rt[a[u]],,N,in[u],ou[u],);
}
else {
x=q[i].x;
int Lca=LCA(u,v);
int res=query(rt[x],,N,in[u]);
res+=query(rt[x],,N,in[v]);
res-=query(rt[x],,N,in[Lca]);
res-=query(rt[x],,N,in[fa[Lca][]]);
printf("%d\n",res);
}
}
return ;
}

BZOJ4999:This Problem Is Too Simple!(DFS序&树上差分&线段树动态开点:区间修改单点查询)的更多相关文章

  1. BZOJ4771 七彩树(dfs序+树上差分+主席树)

    考虑没有深度限制怎么做.显然的做法是直接转成dfs序上主席树,但如果拓展到二维变成矩形数颜色数肯定没法做到一个log. 另一种做法是利用树上差分.对于同种颜色的点,在每个点处+1,dfs序相邻点的lc ...

  2. Codeforces Round #442 (Div. 2) E Danil and a Part-time Job (dfs序加上一个线段树区间修改查询)

    题意: 给出一个具有N个点的树,现在给出两种操作: 1.get x,表示询问以x作为根的子树中,1的个数. 2.pow x,表示将以x作为根的子树全部翻转(0变1,1变0). 思路:dfs序加上一个线 ...

  3. 【bzoj4771】七彩树 树链的并+STL-set+DFS序+可持久化线段树

    题目描述 给定一棵n个点的有根树,编号依次为1到n,其中1号点是根节点.每个节点都被染上了某一种颜色,其中第i个节点的颜色为c[i].如果c[i]=c[j],那么我们认为点i和点j拥有相同的颜色.定义 ...

  4. hdu5692【dfs序】【线段树】

    Snacks Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)Total Sub ...

  5. 【62测试】【状压dp】【dfs序】【线段树】

    第一题: 给出一个长度不超过100只包含'B'和'R'的字符串,将其无限重复下去. 比如,BBRB则会形成 BBRBBBRBBBRB 现在给出一个区间[l,r]询问该区间内有多少个字符'B'(区间下标 ...

  6. 【DFS序】【线段树】bzoj4034 [HAOI2015]T2

    分开维护树的入栈序和出栈序,用两棵线段树.回答时就是用一颗的减去另一棵的. #include<cstdio> #include<algorithm> using namespa ...

  7. 【树链剖分】【dfs序】【线段树】bzoj2836 魔法树

    这道题告诉我们:树链剖分的重标号就是dfs序. #include<cstdio> #include<algorithm> using namespace std; #defin ...

  8. 【BZOJ-3653】谈笑风生 DFS序 + 可持久化线段树

    3653: 谈笑风生 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 628  Solved: 245[Submit][Status][Discuss] ...

  9. BZOJ 3653: 谈笑风生(DFS序+可持久化线段树)

    首先嘛,还是太弱了,想了好久QAQ 然后,这道题么,明显就是求sigma(size[x]) (x是y的儿子且层树小于k) 然后就可以发现:把前n个节点按深度建可持久化线段树,就能用前缀和维护了 其实不 ...

随机推荐

  1. jquery 报错 $.cookie is not a function()

    jquery 报错 $.cookie is not a function() ——我是之前可以运行的项目,突然报这个错误,很奇怪. 这是jquery的cookie插件报错. 插件名: jquery.c ...

  2. python ConfigParse模块(转)

    最近写程序要用到配置文件,那么配置文件的解析就很重要了,下文转自chinaunix 一.ConfigParser简介 ConfigParser 是用来读取配置文件的包.配置文件的格式如下:中括号“[ ...

  3. 01_HBase概述

    1. HBase在Hadoop生态圈中的位置 问题:HBase 是什么,用在哪里,解决什么样的问题? 解答: 1)简单来说, HBase 是一种类似于面向列的分布式数据库(集群), 底层利用HDFS ...

  4. poj 1050 To the Max 最大子矩阵和 经典dp

    To the Max   Description Given a two-dimensional array of positive and negative integers, a sub-rect ...

  5. 前端解析cookie出现多余的双引号的问题

    登录界面,用户提交后,后台获取到用户名密码,然后会设置cookie,以便于前端使用,今天在修改项目记住密码的功能的时候,读取cookie,如果存在loginInfo字段说明用户在上一次点击了记住密码的 ...

  6. 30分钟掌握Dart语言

    在Dart中,一切都是对象,一切对象都是class的实例,哪怕是数字类型.方法甚至null都是对象,所有的对象都是继承自Object 虽然Dart是强类型语言,但变量类型是可选的因为Dart可以自动推 ...

  7. offset的坑 使用前要将对象先show

    使用jquery $('#obj').offset( {top:300, left: 600}); 如果设置offset之前是隐藏的,那么你设置新的offset之后就不会是指定的位置,而且会越飞越远, ...

  8. 最新Dubbo-admin+Zookeeper搭建

    Zookeeper搭建: 下载zookeeper压缩包并解压,下载地址:http://www.apache.org/dyn/closer.cgi/zookeeper/进入conf目录下将 zoo_sa ...

  9. UML中的组合、聚合、关联、继承、实现、依赖

    转自:http://justsee.iteye.com/blog/808799 UML定义的关系主要有六种:依赖.类属.关联.实现.聚合和组合. 继承 指的是一个类(称为子类.子接口)继承另外的一个类 ...

  10. 设计模式--中介者模式C++实现

    中介者模式C++实现 1定义 用一个中介对象封装一系列的对象交互,中介者使各个对象不需要显示的相互作用,从而使其耦合松散,而且可以独立的改变他们之间的交互 2类图 组成说明 Mediator抽象中介者 ...