BZOJ3589 动态树[树剖/暴力/容斥]
操作0,显然直接线段树解决。
操作1,瓶颈在于重叠的链只算一次。在线段树上来看,如果一个区间被覆盖了,那么只算这个区间,子树里面也就不管了。
考虑对节点打标记来表示是否覆盖。但是,如果统一打完之后,并不方便计算打上标记的点的和。明确目标,现在希望能覆盖很多小区间的一个大区间被打上标记之后用他来更新答案。`````
可以对每一个点维护$acc_i$表示这个点子树内被覆盖的区间的和。那么,当有更大的区间覆盖上去的时候,直接把$acc_i$改成$sum_i$,传上去即可,同时在这个点打上已覆盖的标记。否则正常pushup。答案是$acc_{root}$
这样就保证了覆盖后答案只算一次的正确性。
还有一个问题,在操作完之后,显然应当把标记删掉。这里考虑区间覆盖性标记$tag$,当他是1的时候就是之前的完全覆盖,是-1表示没有覆盖,为了使得下一操作时清除$acc$记录,设0表示要把子树内覆盖的区间清掉。
在下放标记的时候,就可以保证左右子树以及根的$acc$都是0,能重新用。当$tag$是1的时候,可以正常下放,也可以不下放,但将0和1统一一下code更简单。复杂度$O(nklog^2n)$
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
#define dbg(x) cerr << #x << " = " << x <<endl
using namespace std;
typedef long long ll;
typedef double db;
typedef pair<int,int> pii;
template<typename T>inline T _min(T A,T B){return A<B?A:B;}
template<typename T>inline T _max(T A,T B){return A>B?A:B;}
template<typename T>inline char MIN(T&A,T B){return A>B?(A=B,):;}
template<typename T>inline char MAX(T&A,T B){return A<B?(A=B,):;}
template<typename T>inline void _swap(T&A,T&B){A^=B^=A^=B;}
template<typename T>inline T read(T&x){
x=;int f=;char c;while(!isdigit(c=getchar()))if(c=='-')f=;
while(isdigit(c))x=x*+(c&),c=getchar();return f?x=-x:x;
}
const int N=2e5+;
struct thxorz{int to,nxt;}G[N<<];
int hd[N],tot;
int n,q;
inline void Addedge(int x,int y){
G[++tot].to=y,G[tot].nxt=hd[x],hd[x]=tot;
G[++tot].to=x,G[tot].nxt=hd[y],hd[y]=tot;
}
#define y G[j].to
int fa[N],d[N],son[N],cnt[N],topfa[N],st[N],tim;
void dfs1(int x,int f){
fa[x]=f,d[x]=d[f]+,cnt[x]=;int tmp=-;
for(register int j=hd[x];j;j=G[j].nxt)if(y^f)dfs1(y,x),cnt[x]+=cnt[y],MAX(tmp,cnt[y])&&(son[x]=y);
}
void dfs2(int x,int topf){
topfa[x]=topf,st[x]=++tim;if(!son[x])return;dfs2(son[x],topf);
for(register int j=hd[x];j;j=G[j].nxt)if(y^fa[x]&&y^son[x])dfs2(y,y);
}
#undef y
#define lc i<<1
#define rc i<<1|1
int sumv[N<<],acc[N<<],stag[N<<],atag[N<<];
inline void Pushup(int i){sumv[i]=sumv[lc]+sumv[rc],acc[i]=acc[lc]+acc[rc];}
inline void pd_sum(int i,int L,int R){
if(stag[i]){
int mid=L+R>>;
sumv[lc]+=(mid-L+)*stag[i],sumv[rc]+=(R-mid)*stag[i],stag[lc]+=stag[i],stag[rc]+=stag[i],stag[i]=;
}
}
inline void pd_acc(int i){
if(~atag[i])
acc[lc]=atag[i]?sumv[lc]:,acc[rc]=atag[i]?sumv[rc]:,atag[rc]=atag[lc]=atag[i],atag[i]=-;
}
void Update_add(int i,int L,int R,int ql,int qr,int k){
if(ql<=L&&qr>=R){sumv[i]+=(R-L+)*k,stag[i]+=k;return;}
int mid=L+R>>;pd_sum(i,L,R);
if(ql<=mid)Update_add(lc,L,mid,ql,qr,k);
if(qr>mid)Update_add(rc,mid+,R,ql,qr,k);
Pushup(i);
}
void Update_account(int i,int L,int R,int ql,int qr){
if(ql<=L&&qr>=R){acc[i]=sumv[i],atag[i]=;return;}
int mid=L+R>>;pd_sum(i,L,R),pd_acc(i);
if(ql<=mid)Update_account(lc,L,mid,ql,qr);
if(qr>mid)Update_account(rc,mid+,R,ql,qr);
Pushup(i);
}
inline void clear_the_tree(){atag[]=,acc[]=;}
inline void Tree_query(int x,int y){
if(d[x]<d[y])x^=y^=x^=y;
while(topfa[x]^topfa[y])Update_account(,,n,st[topfa[x]],st[x]),x=fa[topfa[x]];
Update_account(,,n,st[y],st[x]);
} int main(){//freopen("test.in","r",stdin);freopen("test.ans","w",stdout);
read(n);for(register int i=,x,y;i<n;++i)read(x),read(y),Addedge(x,y);
dfs1(,),dfs2(,);
read(q);memset(atag,-,sizeof atag);
for(register int i=,opt,k,x,y;i<=q;++i){
read(opt);
if(opt){
read(k);while(k--)read(x),read(y),Tree_query(x,y);
printf("%d\n",acc[]&0x7fffffff);clear_the_tree();
}
else read(x),read(k),Update_add(,,n,st[x],st[x]+cnt[x]-,k);
}
return ;
}
还有一个点,对$2^{31}$取模用int自然溢出是因为前31位爆掉,进位到符号位,符号位会变化,但不影响前31位的mod结果。最后只要把符号位改一下即可,所以要$\text{and}2^{31}-1$。
反思:标记的表示设计想的不太好。当线段树设计有限制条件的询问时,可以维护一些附加信息,如本题的覆盖区间的和。
附:本题还有另外两种做法。
一:暴力
每次询问把跳过的每条重链全弄出来,把这些区间按dfs序从小到大排序,然后合并区间,暴力查询。复杂度$O(n(klognlog(klogn)+klog^2n))$。比较卡。
二:容斥
这个询问相当于求若干链的并,可以容斥。
$sum(\bigcup\limits_{i=1}^{n}A_i)=\sum\limits_{k=1}^{n}(-1)^{k-1}\sum\limits_{1<=j_1<j_2<...<j_k<=n}sum(A_{j_1}\cap
A_{j_2}\cap ...\cap A_{j_k})$
参考这篇。
BZOJ3589 动态树[树剖/暴力/容斥]的更多相关文章
- bzoj3589 动态树 求链并 容斥
bzoj3589 动态树 链接 bzoj 思路 求链并. 发现只有最多5条链子,可以容斥. 链交求法:链顶是两条链顶深度大的那个,链底是两个链底的\(lca\) 如果链底深度小于链顶,就说明两条链没有 ...
- hdu 5664 Lady CA and the graph(树的点分治+容斥)
题意: 给你一个有n个点的树,给定根,叫你找第k大的特殊链 .特殊的链的定义:u,v之间的路径,经过题给的根节点. 题解:(来自BC官方题解) 对于求第k大的问题,我们可以通过在外层套一个二分,将其转 ...
- hdu 5792(树状数组,容斥) World is Exploding
hdu 5792 要找的无非就是一个上升的仅有两个的序列和一个下降的仅有两个的序列,按照容斥的思想,肯定就是所有的上升的乘以所有的下降的,然后再减去重复的情况. 先用树状数组求出lx[i](在第 i ...
- Luogu4528 CTSC2008 图腾 树状数组、容斥
传送门 设$f_i$表示$i$排列的数量,其中$x$表示不确定 那么$$ans=f_{1324}-f_{1432}-f_{1243}=(f_{1x2x}-f_{1423})-(f_{14xx}-f_{ ...
- hdu 5792 World is Exploding 树状数组+离散化+容斥
World is Exploding Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Other ...
- Codeforces 439E Devu and Birthday Celebration 容斥
Devu and Birthday Celebration 我们发现不合法的整除因子在 m 的因子里面, 然后枚举m的因子暴力容斥, 或者用莫比乌斯系数容斥. #include<bits/std ...
- BZOJ1853 Scoi2010 幸运数字 【枚举+容斥】
BZOJ1853 Scoi2010 幸运数字 Description 在中国,很多人都把6和8视为是幸运数字!lxhgww也这样认为,于是他定义自己的“幸运号码”是十进制表示中只包含数字6和8的那些号 ...
- 51nod 1486 大大走格子(容斥+dp+组合数)
传送门 解题思路 暴力容斥复杂度太高,无法接受,考虑用\(dp\).设\(f(i)\)表示从左上角开始不经过前面的阻断点,只经过\(i\)的阻断点.那么可以考虑容斥,用经过\(i\)的总方案数减去前面 ...
- bzoj3589 动态树 树链剖分+容斥
题目传送门 https://lydsy.com/JudgeOnline/problem.php?id=3589 题解 事件 \(0\) 不需要说,直接做就可以了. 事件 \(1\) 的话,考虑如果直接 ...
随机推荐
- ######【Python】【基础知识】Python的介绍 ######
Python 是一种面向对象.解释型计算机程序设计语言. Python是什么? Python(英国发音:/ˈpaɪθən/ 美国发音:/ˈpaɪθɑːn/), 是一种面向对象的解释型计算机程序设计语言 ...
- 【LOJ】#3030. 「JOISC 2019 Day1」考试
LOJ#3030. 「JOISC 2019 Day1」考试 看起来求一个奇怪图形(两条和坐标轴平行的线被切掉了一个角)内包括的点个数 too naive! 首先熟练的转化求不被这个图形包含的个数 -- ...
- Linux系列之putty远程登录
在工作中,我们通常都是通过远程操作Linux服务器的,因此必须熟悉一些远程登录的软件,在此使用的是putty,在Windows上安装putty软件,通过该软件访问Linux主机. 1.远程登录步骤 1 ...
- 基于licode搭建webrtc服务器
0. 前言 licode官网文档安装教程十分简单, 但是实际搭建过程是很艰辛的. 官方文档没有提示说会遇到什么样的问题, 实际过程中可能遇到各种各样的问题, 在解决的时候费时费力, 我就总结一下自己在 ...
- 第一次入坑docker
直接进入主题 1.首先获取git clone项目 2.创建镜像:docker build -t="docker" .(注意千万不要忘了.) 3.列出镜像:docker images ...
- 后缀数组练习4:Life Forms
有一个细节不是特别懂,然后的话细节有点多,就是挺难发现的那一种,感谢大佬的博客 1470: 后缀数组4:Life Forms poj3294 时间限制: 1 Sec 内存限制: 128 MB提交: ...
- 网络名称空间 实例研究 veth处于不同网络的路由问题
相关命令详细介绍参见 http://www.cnblogs.com/Dream-Chaser/p/7077105.html .问题: 两个网络名称空间中的两个接口veth0和veth1,如何配置net ...
- List 集合 一行4个排序
List<string> list = new List<string>(); ; i < ; i++) { list.Add(i.ToString()); } int ...
- 记一次配置阿里云ECS GPU计算型gn5实例
基础配置 CPU: Intel(R) Xeon(R) CPU E5-2682 v4 @ 2.50GHz * 16 MEM: 120 GiB GPU: NVIDIA P100 * 2 OS: Ubunt ...
- python与pip
python , pip 相关命令汇总 1) 在python3 下升级pip3 pip3 install --upgrade pip