BZOJ_3589_动态树_容斥原理+树链剖分
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_动态树_容斥原理+树链剖分的更多相关文章
- 树的统计Count---树链剖分
NEFU专项训练十和十一——树链剖分 Description 一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w.我们将以下面的形式来要求你对这棵树完成一些操作: I. CHANGE u t ...
- [LOJ3014][JOI 2019 Final]独特的城市——树的直径+长链剖分
题目链接: [JOI 2019 Final]独特的城市 对于每个点,它的答案最大就是与它距离最远的点的距离. 而如果与它距离为$x$的点有大于等于两个,那么与它距离小于等于$x$的点都不会被计入答案. ...
- BZOJ 1036: [ZJOI2008]树的统计Count-树链剖分(点权)(单点更新、路径节点最值、路径求和)模板,超级认真写了注释啊啊啊
1036: [ZJOI2008]树的统计Count Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 23015 Solved: 9336[Submit ...
- [bzoj2733][HNOI2012]永无乡_权值线段树_线段树合并
永无乡 bzoj-2733 HNOI-2012 题目大意:题目链接. 注释:略. 想法: 它的查询操作非常友善,就是一个联通块内的$k$小值. 故此我们可以考虑每个联通块建一棵权值线段树. 这样的话每 ...
- 3065: 带插入区间K小值_树套树_替罪羊树_权值线段树
经过周六一天,周一3个小时的晚自习,周二2个小时的疯狂debug,终于凭借自己切掉了这道树套树题. Code: #include <cstdio> #include <algorit ...
- 219.01.19 bzoj3252: 攻略(长链剖分+贪心)
传送门 长链剖分好题. 题意:给一棵带点权的树,可以从根节点到任一叶节点走kkk次,走过的点只能计算一次,问kkk次走过的点点权值和最大值. 思路: 考虑将整棵树带权长链剖分,这样链与链之间是不会重复 ...
- 【BZOJ-3589】动态树 树链剖分 + 线段树 + 线段覆盖(特殊的技巧)
3589: 动态树 Time Limit: 30 Sec Memory Limit: 1024 MBSubmit: 405 Solved: 137[Submit][Status][Discuss] ...
- B20J_3231_[SDOI2014]旅行_树链剖分+线段树
B20J_3231_[SDOI2014]旅行_树链剖分+线段树 题意: S国有N个城市,编号从1到N.城市间用N-1条双向道路连接,城市信仰不同的宗教,为了方便,我们用不同的正整数代表各种宗教. S国 ...
- [bzoj 3531][SDOI2014]旅行(树链剖分+动态开点线段树)
题目:http://www.lydsy.com:808/JudgeOnline/problem.php?id=3531 分析: 对于每个颜色(颜色<=10^5)都建立一颗线段树 什么!那么不是M ...
随机推荐
- 关于Linux和Unix的分析
Linux操作系统即linux. Linux是一套免费使用和自由传播的类Unix操作系统,是一个基于POSIX和UNIX的多用户.多任务.支持多线程和多CPU的操作系统.它能运行主要的UNIX工具软 ...
- 自动生成MyEclipse 安装破解码
新建一个class 文件,Debug 模式运行一个,输入任意值 ,回车得到破解安装码 代码文件如下: import java.io.*; public class MyEclipseGen { pri ...
- Angular HttpClient upload file with FormData
从sof上找到一个example:https://stackoverflow.com/questions/46206643/asp-net-core-2-0-and-angular-4-3-file- ...
- Django push: Using Server-Sent Events and WebSocket with Django
http://curella.org/blog/2012/jul/17/django-push-using-server-sent-events-and-websocket/ The goal of ...
- java之Spring(IOC)装配Bean(手动装配、自动装配、注解装配)
在上一篇控制反转中我们看到了依靠一个Bean文件来实现对代码的控制,可谓十分便捷,再也不用去实例化对象了,2333~~~ 1.手动装配 <bean id="todo" cla ...
- python爬虫入门(九)Scrapy框架之数据库保存
豆瓣电影TOP 250爬取-->>>数据保存到MongoDB 豆瓣电影TOP 250网址 要求: 1.爬取豆瓣top 250电影名字.演员列表.评分和简介 2.设置随机UserAge ...
- 用ASP.NET Core 2.0 建立规范的 REST API -- 预备知识 (2) + 准备项目
上一部分预备知识在这 http://www.cnblogs.com/cgzl/p/9010978.html 如果您对ASP.NET Core很了解的话,可以不看本文, 本文基本都是官方文档的内容. A ...
- arcEngine开发之查看属性表
这篇文章给出实现属性表功能的具体步骤,之后再对这些步骤中的代码进行分析. 环境准备 拖动TOCControl.MapControl控件到Form窗体上,然后拖动ContextMenuStrip控件至T ...
- springboot集成rabbitmq(实战)
RabbitMQ简介RabbitMQ使用Erlang语言开发的开源消息队列系统,基于AMQP协议来实现(AMQP的主要特征是面向消息.队列.路由.可靠性.安全).支持多种客户端,如:Python.Ru ...
- RocketMQ部分数据消费不了问题排查
问题现象 今天忽然收到RocketMQ预警信息如下: 提醒有部分数据没有消费,产生堆积情况. 打开RocketMq-Console-Ng查看如下图形式: 备注:第一反应是Consumer Group内 ...