BZOJ_3589_动态树_容斥原理+树链剖分

题意:

维护一棵树,支持1.子树内点权加上一个数  2.给出k条链,求路径上的点权和(重复的计算一次) (k<=5)

分析:

可以用树剖+线段树解决第一个操作

然后我们发现k非常小,可以二进制枚举

那就容斥一下转化成求几条链的交

链交求法:链顶是两条链顶深度大的那个,链底是两个链底的lca

如果链底深度小于链顶,就说明两条链没有交集

代码:

#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
#define N 200050
#define LL long long
#define ls p<<1
#define rs p<<1|1
int head[N],to[N<<1],nxt[N<<1],cnt,n,q,xx[10],yy[10];
int fa[N],dep[N],top[N],siz[N],son[N],idx[N],tot,k;
int _count[100],strtop[10],strbot[10];
LL mod=2147483648ll;
LL t[N<<2],add[N<<2];
inline void adde(int u,int v){
to[++cnt]=v;nxt[cnt]=head[u];head[u]=cnt;
}
void dfs1(int x,int y){
int i;
fa[x]=y;dep[x]=y+1;
siz[x]=1;
for(i=head[x];i;i=nxt[i]){
if(to[i]!=y){
dfs1(to[i],x);
siz[x]+=siz[to[i]];
if(siz[to[i]]>siz[son[x]])son[x]=to[i];
}
}
}
void dfs2(int x,int t){
top[x]=t;idx[x]=++tot;
int i;
if(son[x]) dfs2(son[x],t);
for(i=head[x];i;i=nxt[i]){
if(to[i]!=fa[x]&&to[i]!=son[x]){
dfs2(to[i],to[i]);
}
}
}
void pud(int l,int r,int p){
if(add[p]==0)return ;
add[ls]+=add[p];
add[ls]%=mod;
add[rs]+=add[p];
add[rs]%=mod;
int mid=l+r>>1;
t[ls]+=add[p]*(mid-l+1);
t[ls]%=mod;
t[rs]+=add[p]*(r-mid);
t[rs]%=mod;
add[p]=0;
}
void update(int l,int r,int x,int y,int c,int p){
if(x<=l&&y>=r){
t[p]+=1ll*c*(r-l+1);
add[p]+=c;
t[p]%=mod;
add[p]%=mod;
return ;
}
pud(l,r,p);
int mid=l+r>>1;
if(x<=mid)update(l,mid,x,y,c,ls);
if(y>mid)update(mid+1,r,x,y,c,rs);
t[p]=t[ls]+t[rs];
t[p]%=mod;
}
LL query(int l,int r,int x,int y,int p){
if(x<=l&&y>=r) return t[p];
int mid=l+r>>1;
LL re=0;
pud(l,r,p);
t[p]=t[ls]+t[rs];
t[p]%=mod;
if(x<=mid)re=(re+query(l,mid,x,y,ls))%mod;
if(y>mid)re=(re+query(mid+1,r,x,y,rs))%mod;
return re%mod;
}
LL ask(int x,int y){
LL re=0;
while(top[x]!=top[y]){
if(dep[top[x]]>dep[top[y]])swap(x,y);
re+=query(1,n,idx[top[y]],idx[y],1);
re%=mod;
y=fa[top[y]];
}
if(dep[x]<dep[y])swap(x,y);
return (re+query(1,n,idx[y],idx[x],1))%mod;
}
void fix(int x,int c){
update(1,n,idx[x],idx[x]+siz[x]-1,c,1);
}
int lca(int x,int y){
while(top[x]!=top[y]){
if(dep[top[x]]>dep[top[y]])swap(x,y);
y=fa[top[y]];
}
return dep[x]<dep[y]?x:y;
}
void solve(){
int mask=(1<<k)-1;
int i,flg,j;
LL ans=0;
for(i=1;i<=k;i++){
if(dep[xx[i]]>dep[yy[i]])swap(xx[i],yy[i]);
strtop[i]=xx[i];strbot[i]=yy[i];
}
for(i=1;i<=mask;i++){
if((_count[i]&1))flg=1;
else flg=-1;
int no_jiao=0;
int top_a=0,bot_a=0;
for(j=1;j<=k;j++){
if(i&(1<<j-1)){
if(!top_a){
top_a=strtop[j];
bot_a=strbot[j];
}
else {
bot_a=lca(bot_a,strbot[j]);
if(dep[top_a]<dep[strtop[j]]){
top_a=strtop[j];
}
if(dep[top_a]>dep[bot_a])no_jiao=1;
}
}
}
if(no_jiao)continue;
ans+=flg*ask(top_a,bot_a);
ans=(ans+mod)%mod;
}
printf("%lld\n",ans);
}
int main(){
scanf("%d",&n);
int i,x,y,opt,j;
for(i=1;i<n;i++){
scanf("%d%d",&x,&y);
adde(x,y);adde(y,x);
}
dfs1(1,0);
dfs2(1,1);
for(i=1;i<=32;i++){
_count[i]=_count[i>>1]+(i&1);
}
scanf("%d",&q);
while(q--){
scanf("%d",&opt);
if(opt){
scanf("%d",&k);
for(j=1;j<=k;j++){
scanf("%d%d",&xx[j],&yy[j]);
}
solve();
}else{
scanf("%d%d",&x,&y);
fix(x,y);
}
}
}

BZOJ_3589_动态树_容斥原理+树链剖分的更多相关文章

  1. 树的统计Count---树链剖分

    NEFU专项训练十和十一——树链剖分 Description 一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w.我们将以下面的形式来要求你对这棵树完成一些操作: I. CHANGE u t ...

  2. [LOJ3014][JOI 2019 Final]独特的城市——树的直径+长链剖分

    题目链接: [JOI 2019 Final]独特的城市 对于每个点,它的答案最大就是与它距离最远的点的距离. 而如果与它距离为$x$的点有大于等于两个,那么与它距离小于等于$x$的点都不会被计入答案. ...

  3. BZOJ 1036: [ZJOI2008]树的统计Count-树链剖分(点权)(单点更新、路径节点最值、路径求和)模板,超级认真写了注释啊啊啊

    1036: [ZJOI2008]树的统计Count Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 23015  Solved: 9336[Submit ...

  4. [bzoj2733][HNOI2012]永无乡_权值线段树_线段树合并

    永无乡 bzoj-2733 HNOI-2012 题目大意:题目链接. 注释:略. 想法: 它的查询操作非常友善,就是一个联通块内的$k$小值. 故此我们可以考虑每个联通块建一棵权值线段树. 这样的话每 ...

  5. 3065: 带插入区间K小值_树套树_替罪羊树_权值线段树

    经过周六一天,周一3个小时的晚自习,周二2个小时的疯狂debug,终于凭借自己切掉了这道树套树题. Code: #include <cstdio> #include <algorit ...

  6. 219.01.19 bzoj3252: 攻略(长链剖分+贪心)

    传送门 长链剖分好题. 题意:给一棵带点权的树,可以从根节点到任一叶节点走kkk次,走过的点只能计算一次,问kkk次走过的点点权值和最大值. 思路: 考虑将整棵树带权长链剖分,这样链与链之间是不会重复 ...

  7. 【BZOJ-3589】动态树 树链剖分 + 线段树 + 线段覆盖(特殊的技巧)

    3589: 动态树 Time Limit: 30 Sec  Memory Limit: 1024 MBSubmit: 405  Solved: 137[Submit][Status][Discuss] ...

  8. B20J_3231_[SDOI2014]旅行_树链剖分+线段树

    B20J_3231_[SDOI2014]旅行_树链剖分+线段树 题意: S国有N个城市,编号从1到N.城市间用N-1条双向道路连接,城市信仰不同的宗教,为了方便,我们用不同的正整数代表各种宗教. S国 ...

  9. [bzoj 3531][SDOI2014]旅行(树链剖分+动态开点线段树)

    题目:http://www.lydsy.com:808/JudgeOnline/problem.php?id=3531 分析: 对于每个颜色(颜色<=10^5)都建立一颗线段树 什么!那么不是M ...

随机推荐

  1. aside元素

    aside元素用来表示当前页面或文章的附属信息部分,它可以包含与当前页面或主要内容相关的引用.侧边栏.广告.导航条,以及其他类似的有别于主要内容的部分. aside元素主要有以下两种使用方法: 1.包 ...

  2. HashMap 深入分析

    /**     *@author annegu     *@date 2009-12-02     */ Hashmap是一种非常常用的.应用广泛的数据类型,最近研究到相关的内容,就正好复习一下.网上 ...

  3. 教你一步步发布一个开源库到 JCenter

    今天想来分享下,如何一步步自己发布一个开源库到 JCenter 这方面的博客网上已经特别多了,所以本篇并不打算仅仅只是记录流程步骤而已,而是尽可能讲清楚,为什么需要有这个步骤,让大伙知其然的同时还知其 ...

  4. html5 标签在 IE 下使用

    (function(){if(!/*@cc_on!@*/0)return;var e = "abbr,article,aside,audio,bb,canvas,datagrid,datal ...

  5. Java接口和抽象类以及接口的意义,instanceof的利用

    接口interface: 1. 在接口中没有变量,成员无论如何定义,都是公共常量,public static final即使不显式声明也如此. 2. 所有接口方法均隐含public abstract即 ...

  6. 2018山东省ACM省赛G题-Game

    Alice and Bob are playing a stone game. There are n piles of stones. In each turn, a player can remo ...

  7. Tomcat PermGen space的解决方案

    Tomcat报告 Caused by: java.lang.OutOfMemoryError: PermGen space异常 内存溢出PermGen space的全称是Permanent Gener ...

  8. Java面向对象(一、封装)

    Java 封装 封装的概念 在面向对象程式设计方法中,封装(英语:Encapsulation)是指一种将抽象性函式接口的实现细节部份包装.隐藏起来的方法. 封装可以被认为是一个保护屏障,防止该类的代码 ...

  9. vue的生命周期钩子

    生命周期过程: new vue() :vue实例进行初始化,读取所有生命周期函数,并没有执行(不会调用) beforeCreate():创建前,读取属性,计算属性,添加set/get,读取watch ...

  10. VS2010+OpenMP的简单使用

    OpenMP是把程序中的循环操作分给电脑的各个CPU处理器并行进行.比如说我要循环运行100次,我的电脑有两个处理器,那OpenMP就会平均分给两个处理器并行运行,每个处理器运行50次.使用方法 1. ...