[jzoj 3175] 数树数 解题报告 (树链剖分)
interlinkage:
https://jzoj.net/senior/#main/show/3175
description:
给定一棵N 个节点的树,标号从1~N。每个点有一个权值。要求维护两种操作:
1. C i x(0<=x<2^31) 表示将i 点权值变为x
2. Q i j x(0<=x<2^31) 表示询问i 到j 的路径上有多少个值为x的节点
solution:
- 链剖
- 把颜色离散化,对每种颜色分别搞一颗线段树
- 直接搞会炸空间,因此要动态开点
- 感觉树上莫队好像也可以
code
#include<iostream>
#include<cstring>
#include<cstdio>
#include<algorithm>
#include<queue>
using namespace std;
typedef long long ll; const int N=4e5+;
int n,q,tot,cnt,tim;
int head[N],dep[N],siz[N],wson[N],fa[N],top[N],dfn[N],rt[N<<],color[N];
int lx[N<<],rx[N<<],sum[N<<];
ll a[N],b[N<<];
struct EDGE
{
int to,nxt;
}edge[N];
struct QUE
{
int op;
int x,y;
ll v;
}t[N];
inline ll read()
{
char ch=getchar();ll s=,f=;
while (ch<''||ch>'') {if (ch=='-') f=-;ch=getchar();}
while (ch>=''&&ch<='') {s=(s<<)+(s<<)+ch-'';ch=getchar();}
return s*f;
}
void add(int u,int v)
{
edge[++tot]=(EDGE){v,head[u]};
head[u]=tot;
}
void dfs1(int x,int pre)
{
dep[x]=dep[pre]+;siz[x]=;fa[x]=pre;
for (int i=head[x];i;i=edge[i].nxt)
{
int y=edge[i].to;
if (y==pre) continue;
dfs1(y,x);
siz[x]+=siz[y];
if (!wson[x]||siz[wson[x]]<siz[y]) wson[x]=y;
}
}
void dfs2(int x,int tp)
{
top[x]=tp;dfn[x]=++tim;
if (wson[x]) dfs2(wson[x],tp);
for (int i=head[x];i;i=edge[i].nxt)
{
int y=edge[i].to;
if (y==fa[x]||y==wson[x]) continue;
dfs2(y,y);
}
}
void upd(int o)
{
sum[o]=sum[lx[o]]+sum[rx[o]];
}
void update(int &o,int l,int r,int pos,int x)
{
if (!o) o=++cnt;
if (l==r)
{
sum[o]+=x;
return;
}
int mid=l+r>>;
if (pos<=mid) update(lx[o],l,mid,pos,x);
else update(rx[o],mid+,r,pos,x);
upd(o);
}
void change(int x,int v)
{
if (color[x])
{
update(rt[color[x]],,n,dfn[x],-);
}
color[x]=v;
update(rt[v],,n,dfn[x],);
}
int query(int o,int l,int r,int x,int y)
{
if (!o) return ;
if (l>=x&&r<=y) return sum[o];
int mid=l+r>>,re=;
if (x<=mid) re+=query(lx[o],l,mid,x,y);
if (y>mid) re+=query(rx[o],mid+,r,x,y);
return re;
}
int query_path(int x,int y,int v)
{
int res=;
while (top[x]!=top[y])
{
if (dep[top[x]]<dep[top[y]]) swap(x,y);
res+=query(rt[v],,n,dfn[top[x]],dfn[x]);
x=fa[top[x]];
}
if (dep[x]<dep[y]) swap(x,y);
res+=query(rt[v],,n,dfn[y],dfn[x]);
return res;
}
int main()
{
//freopen("tree.in","r",stdin);
n=read();q=read();
int len=;
for (int i=;i<=n;i++) a[i]=read(),b[++len]=a[i];
for (int i=;i<n;i++)
{
int u=read(),v=read();
add(u,v);add(v,u);
}
dfs1(,);
dfs2(,);
char s[];
for (int i=;i<=q;i++)
{
scanf("%s",s);
if (s[]=='C')
{
t[i].op=;
t[i].x=read();t[i].v=read();
}
if (s[]=='Q')
{
t[i].op=;
t[i].x=read();t[i].y=read();t[i].v=read();
}
b[++len]=t[i].v;
}
sort(b+,b++len);
len=unique(b+,b++len)-b-;
for (int i=;i<=n;i++) a[i]=lower_bound(b+,b++len,a[i])-b;
for (int i=;i<=q;i++) t[i].v=lower_bound(b+,b++len,t[i].v)-b;
for (int i=;i<=n;i++) change(i,a[i]);
for (int i=;i<=q;i++)
{
if (t[i].op==)
{
change(t[i].x,t[i].v);
}
if (t[i].op==)
{
printf("%d\n",query_path(t[i].x,t[i].y,t[i].v));
}
}
return ;
}
[jzoj 3175] 数树数 解题报告 (树链剖分)的更多相关文章
- 【九度OJ】题目1176:树查找 解题报告
[九度OJ]题目1176:树查找 解题报告 标签(空格分隔): 九度OJ http://ac.jobdu.com/problem.php?pid=1176 题目描述: 有一棵树,输出某一深度的所有节点 ...
- (持续更新)虚树,KD-Tree,长链剖分,后缀数组,后缀自动机
真的就是讲课两天,吸收一个月呢! \(1.\)虚树 \(2.\)KD-Tree \(3.\)长链剖分 \(4.\)后缀数组 后缀数组 \(5.\)后缀自动机 后缀自动机
- [BZOJ2243][SDOI2011]染色 解题报告|树链剖分
Description 给定一棵有n个节点的无根树和m个操作,操作有2类: 1.将节点a到节点b路径上所有点都染成颜色c: 2.询问节点a到节点b路径上的颜色段数量(连续相同颜色被认为是同一段),如“ ...
- [BZOJ1036][ZJOI2008]树的统计Count 解题报告|树链剖分
树链剖分 简单来说就是数据结构在树上的应用.常用的为线段树splay等.(可现在splay还不会敲囧) 重链剖分: 将树上的边分成轻链和重链. 重边为每个节点到它子树最大的儿子的边,其余为轻边. 设( ...
- [POJ3237]Tree解题报告|树链剖分|边剖
关于边剖 之前做的大多是点剖,其实转换到边剖非常简单. 我的做法是每个点的点权记录其到父亲节点的边的边权. 只要solve的时候不要把最上面的点记录在内就可以了. Tree Description Y ...
- HDU 1754 线段树入门解题报告
---恢复内容开始--- 题意:给定区间,每个人的成绩, Q次询问,求每次询问区间中的最大值 思路:构造线段树 代码: #include<stdio.h> #include<algo ...
- POJ 3264 线段树入门解题报告
题意:给n个值, Q次询问, 每次询问给定一个区间, 要求输出该区间最大最小值之差 思路:暴力的话每次询问都要遍历多次for循环一定会超时, 用线段树记录区间的信息(左边界右边界, 该区间最大值最小值 ...
- [NOIP2016 DAY1 T2]天天爱跑步-[差分+线段树合并][解题报告]
[NOIP2016 DAY1 T2]天天爱跑步 题面: B[NOIP2016 DAY1]天天爱跑步 时间限制 : - MS 空间限制 : 565536 KB 评测说明 : 2s Description ...
- 洛谷 P3373 【模板】线段树 2 解题报告
P3373 [模板]线段树 2 题目描述 如题,已知一个数列,你需要进行下面三种操作: 1.将某区间每一个数乘上\(x\) 2.将某区间每一个数加上\(x\) 3.求出某区间每一个数的和 输入输出格式 ...
随机推荐
- [lua] future模式*协程
以下是lua实现的future模式.基于cocos客户端 local function param_pack( params, callback ) table.insert(params, call ...
- sql server 创建索引 超时时间已到
如下图所示:在现场PR_Product表中添加绯聚焦索引PSCode,点击保存按钮后等了一段时间弹出超时警告!现场这张表的数据量也是特别大的(250+万),但是我本地也是把现场数据库还原了的,一样的数 ...
- Android一对多蓝牙连接示例APP
一对多蓝牙连接示例,基于Google BluetoothChat修改,实现一对多聊天(一个服务端.多个客户端),类似聊天室. 主要功能: 客户端的发出的消息所有终端都能收到(由服务端转发) 客户端之间 ...
- Zabbix4.0 Web管理界面中文乱码解决方法(转)
Zabbix安装好之后,监控图形页面出现字符集乱码 解决方法:1.复制本地电脑C:\Windows\Fonts\simkai.ttf(楷体)上传到zabbix服务器网站目录的fonts目录下 2.za ...
- vue 配置页面动态的 title
- BZOJ 3158 千钧一发 (最大流->二分图带权最大独立集)
题面:BZOJ传送门 和方格取数问题很像啊 但这道题不能像网格那样黑白染色构造二分图,所以考虑拆点建出二分图 我们容易找出数之间的互斥关系,在不能同时选的两个点之间连一条流量为$inf$的边 由于我们 ...
- [网络流24题] 太空飞行计划问题 (最大流->最大权闭合图)
洛谷传送门 LOJ传送门 做这道题之前建议先看这篇论文,虽然论文里很多地方用了很多术语,但hbt神犇讲得很明白 这篇题解更加偏向于感性理解 把问题放到二分图上,左侧一列点是实验,权值为$p[i]$,右 ...
- 酒店管理系统(功能结构图、E-R图、用例图)
功能结构图 用例图 状态图 流程图 数据字典 er图
- SpringBoot-CommandLineRunner实现预操作
前提:在使用SpringBoot构建项目时,我们通常需要做一些预先操作(类似开机自启动).而SpringBoot正好提供了一个简单的方式来实现–CommandLineRunner. CommandLi ...
- JavaScript 事件对内存和性能的影响
程序代码: <%-- Created by IntelliJ IDEA. User: 乔克叔叔 Date: 2017/12/26 Time: 16:45 To change this templ ...