利用树剖序的一些性质~

这个题可以出到 $\sum k=10^5$ 左右.
做法很简单:每次暴力跳重链,并在线段树上查询链和.
查询之后打一个标记,把加过的链都置为 $0$.
这样的话在同一次询问时即使有重复的也无所谓.
然后本次查询后在线段树的 $1$ 号节点再打一个标记,用来将那些置零的标记清空.
这个主要是利用树剖序的连续性质.
具体细节还是需要注意一下.

Code:

#include <cstdio>
#include <algorithm>
#define N 200004
#define ll long long
#define setIO(s) freopen(s".in","r",stdin) , freopen(s".out","w",stdout)
using namespace std;
const ll mod=2147483648;
void Add(ll &g,ll b)
{
g=(g+b)%mod;
}
namespace seg
{
struct Node
{
int tag,mark;
ll sum,sumvir,add;
}t[N<<2];
void mark_tag(int now)
{
t[now].mark=1,t[now].tag=0,t[now].sum=t[now].sumvir;
}
void mark_add(int l,int r,int now,ll v)
{
Add(t[now].add,v), Add(t[now].sumvir,(r-l+1)*v%mod);
if(t[now].tag) t[now].sum=0;
else t[now].sum=t[now].sumvir;
}
void pushdown(int l,int r,int now)
{
int mid=(l+r)>>1;
if(t[now].mark)
{
if(mid>=l) mark_tag(now<<1);
if(r>mid) mark_tag(now<<1|1);
t[now].mark=0;
}
if(t[now].add)
{
if(mid>=l) mark_add(l,mid,now<<1,t[now].add);
if(r>mid) mark_add(mid+1,r,now<<1|1,t[now].add);
t[now].add=0;
}
}
void pushup(int l,int r,int now)
{
int mid=(l+r)>>1;
t[now].sum=t[now].sumvir=0;
if(mid>=l) Add(t[now].sum,t[now<<1].sum),Add(t[now].sumvir,t[now<<1].sumvir);
if(r>mid) Add(t[now].sum,t[now<<1|1].sum),Add(t[now].sumvir,t[now<<1|1].sumvir);
if(t[now].tag) t[now].sum=0;
}
void update_tag(int l,int r,int now,int L,int R)
{
if(l>=L&&r<=R)
{
t[now].tag=1,t[now].sum=t[now].mark=0;
return;
}
pushdown(l,r,now);
int mid=(l+r)>>1;
if(L<=mid) update_tag(l,mid,now<<1,L,R);
if(R>mid) update_tag(mid+1,r,now<<1|1,L,R);
pushup(l,r,now);
}
void update_add(int l,int r,int now,int L,int R,ll v)
{
if(l>=L&&r<=R)
{
mark_add(l,r,now,v);
return;
}
pushdown(l,r,now);
int mid=(l+r)>>1;
if(L<=mid) update_add(l,mid,now<<1,L,R,v);
if(R>mid) update_add(mid+1,r,now<<1|1,L,R,v);
pushup(l,r,now);
}
ll query(int l,int r,int now,int L,int R)
{
if((l>=L&&r<=R)||(t[now].tag)) return t[now].sum;
int mid=(l+r)>>1;
pushdown(l,r,now);
ll re=0;
if(L<=mid) Add(re,query(l,mid,now<<1,L,R));
if(R>mid) Add(re,query(mid+1,r,now<<1|1,L,R));
return re;
}
};
int n,Q,edges,tim;
int hd[N],to[N<<1],nex[N<<1],val[N];
int size[N],son[N],top[N],dfn[N],fa[N],dep[N],st[N],ed[N];
void add(int u,int v)
{
nex[++edges]=hd[u],hd[u]=edges,to[edges]=v;
}
void dfs1(int u,int ff)
{
fa[u]=ff,dep[u]=dep[ff]+1,size[u]=1;
for(int i=hd[u];i;i=nex[i])
if(to[i]!=ff)
{
dfs1(to[i],u),size[u]+=size[to[i]];
if(size[to[i]]>size[son[u]]) son[u]=to[i];
}
}
void dfs2(int u,int tp)
{
top[u]=tp,st[u]=dfn[u]=++tim;
if(son[u]) dfs2(son[u],tp);
for(int i=hd[u];i;i=nex[i])
if(to[i]!=fa[u]&&to[i]!=son[u]) dfs2(to[i],to[i]);
ed[u]=tim;
}
ll work(int x,int y)
{
ll re=0;
while(top[x]!=top[y])
{
if(dep[top[y]]>dep[top[x]]) swap(x,y);
Add(re,seg::query(1,n,1,dfn[top[x]],dfn[x]));
seg::update_tag(1,n,1,dfn[top[x]],dfn[x]);
x=fa[top[x]];
}
if(dep[x]<dep[y]) swap(x,y);
Add(re,seg::query(1,n,1,dfn[y],dfn[x]));
seg::update_tag(1,n,1,dfn[y],dfn[x]);
return re;
}
int main()
{
int i,j;
// setIO("input");
scanf("%d",&n);
for(i=1;i<n;++i)
{
int u,v;
scanf("%d%d",&u,&v),add(u,v),add(v,u);
}
dfs1(1,0),dfs2(1,1);
scanf("%d",&Q);
for(int cas=1;cas<=Q;++cas)
{
int opt;
scanf("%d",&opt);
if(opt==0)
{
int u;
ll v;
scanf("%d%lld",&u,&v);
seg::update_add(1,n,1,st[u],ed[u],v);
}
if(opt==1)
{
ll ans=0;
int k,u,v;
scanf("%d",&k);
for(i=1;i<=k;++i) scanf("%d%d",&u,&v), Add(ans,work(u,v));
seg::mark_tag(1);
printf("%lld\n",ans);
}
}
return 0;
}

  

BZOJ 3589: 动态树 树链剖分+线段树+树链的并的更多相关文章

  1. BZOJ 3589 动态树 (树链剖分+线段树)

    前言 众所周知,90%90\%90%的题目与解法毫无关系. 题意 有一棵有根树,两种操作.一种是子树内每一个点的权值加上一个同一个数,另一种是查询多条路径的并的点权之和. 分析 很容易看出是树链剖分+ ...

  2. BZOJ 3531 SDOI2014 旅行 树链剖分+线段树动态开点

    题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=3531 题意概述: 给出一棵N个点的树,树上的每个结点有一个颜色和权值,支持以下四种操作: ...

  3. BZOJ.1758.[WC2010]重建计划(分数规划 点分治 单调队列/长链剖分 线段树)

    题目链接 BZOJ 洛谷 点分治 单调队列: 二分答案,然后判断是否存在一条长度在\([L,R]\)的路径满足权值和非负.可以点分治. 对于(距当前根节点)深度为\(d\)的一条路径,可以用其它子树深 ...

  4. 【bzoj5210】最大连通子块和 树链剖分+线段树+可删除堆维护树形动态dp

    题目描述 给出一棵n个点.以1为根的有根树,点有点权.要求支持如下两种操作: M x y:将点x的点权改为y: Q x:求以x为根的子树的最大连通子块和. 其中,一棵子树的最大连通子块和指的是:该子树 ...

  5. 【bzoj4712】洪水 树链剖分+线段树维护树形动态dp

    题目描述 给出一棵树,点有点权.多次增加某个点的点权,并在某一棵子树中询问:选出若干个节点,使得每个叶子节点到根节点的路径上至少有一个节点被选择,求选出的点的点权和的最小值. 输入 输入文件第一行包含 ...

  6. BZOJ.4034 [HAOI2015]树上操作 ( 点权树链剖分 线段树 )

    BZOJ.4034 [HAOI2015]树上操作 ( 点权树链剖分 线段树 ) 题意分析 有一棵点数为 N 的树,以点 1 为根,且树点有边权.然后有 M 个 操作,分为三种: 操作 1 :把某个节点 ...

  7. BZOJ.1036 [ZJOI2008]树的统计Count ( 点权树链剖分 线段树维护和与最值)

    BZOJ.1036 [ZJOI2008]树的统计Count (树链剖分 线段树维护和与最值) 题意分析 (题目图片来自于 这里) 第一道树链剖分的题目,谈一下自己的理解. 树链剖分能解决的问题是,题目 ...

  8. P3313 [SDOI2014]旅行——树链剖分+线段树(动态开点?)

    P3313 [SDOI2014]旅行 一棵树,其中的点分类,点有权值,在一条链上找到一类点中的最大值或总和: 树链剖分把树变成链: 把每个宗教单开一个线段树,维护区间总和和最大值: 宗教很多,需要动态 ...

  9. BZOJ 3672[NOI2014]购票(树链剖分+线段树维护凸包+斜率优化) + BZOJ 2402 陶陶的难题II (树链剖分+线段树维护凸包+分数规划+斜率优化)

    前言 刚开始看着两道题感觉头皮发麻,后来看看题解,发现挺好理解,只是代码有点长. BZOJ 3672[NOI2014]购票 中文题面,题意略: BZOJ 3672[NOI2014]购票 设f(i)f( ...

  10. 洛谷P3313 [SDOI2014]旅行 题解 树链剖分+线段树动态开点

    题目链接:https://www.luogu.org/problem/P3313 这道题目就是树链剖分+线段树动态开点. 然后做这道题目之前我们先来看一道不考虑树链剖分之后完全相同的线段树动态开点的题 ...

随机推荐

  1. PTA(Basic Level)1057.数零壹

    给定一串长度不超过 105 的字符串,本题要求你将其中所有英文字母的序号(字母 a-z 对应序号 1-26,不分大小写)相加,得到整数 N,然后再分析一下 N 的二进制表示中有多少 0.多少 1.例如 ...

  2. IIS7多站点ssl配置及http自动跳转到https

    SSL证书配置参考如下: http转https实战教程iis7.5 window08 IIS7安装多域名SSL证书绑定443端口 关键是修改C:\Windows\System32\inetsrv\co ...

  3. windows10 AppStore安装 应用商店重新安装

    点击左下角的搜索按钮,如下图所示   输入powershell,在结果中找到widows powershell应用,如下图所示   右键单击widows powershell应用,选择以管理员运行,如 ...

  4. Markedown换行

    1.换行 1⃣️行尾先加两个空格符后再按回车键,显示内容就会换行 2⃣️行尾连按两次回车键,显示内容就会换行 3⃣️举例: 第一行内容<br/>第二行内容 第一行内容<br>第 ...

  5. BugkuCTF--never give up

    这道题还挺有意思的... http://123.206.87.240:8006/test/hello.php 查看元素,有个1p.html,访问. 还没看到网页元素就跳转了...抓包! 抓到了一堆东西 ...

  6. python-day40(正式学习)

    目录 线程队列 1 2 3 线程定时器 进程池和线程池 线程队列 1 import queue q=queue.Queue() q.put('123') q.put('456') q.put('789 ...

  7. Redis安全策略

    1. 开启redis密码认证,并设置高复杂度密码 描述 redis在redis.conf配置文件中,设置配置项requirepass, 开户密码认证. redis因查询效率高,auth这种命令每秒能处 ...

  8. Dubbo架构

    原文链接http://dubbo.apache.org 架构图 节点角色说明 节点 角色说明 Provider 暴露服务的服务提供方 Consumer 调用远程服务的服务消费方 Registry 服务 ...

  9. Express multer 文件上传

    npm multer 文件上传 Express app 范本就不写了,仅记录一下上传部分的代码. const fs = require('fs'); const express = require(' ...

  10. vue路由守卫触发顺序

    不同组件之间的路由跳转流程图 导航被触发(A–>B) 调用A组件内路由守卫beforeRouteLeave(to,from,next) 调用全局路由前置守卫router.beforeEach(t ...