BZOJ 4855 [Jsoi2016]轻重路径
题解:用树链剖分来维护树链剖分
令d[x]=size[heavyson[x]]-size[lightson[x]]
当d[x]<0时轻重儿子关系改变
用数据结构维护d[x]并找到这些位置改变即可
时间复杂度O(不会分析)
#include<iostream>
#include<cstdio>
#include<cstring>
#define lo now<<1
#define ro now<<1|1
using namespace std;
const int maxn=200009; int n,m; long long nowans=0;
int havein[maxn],root=0;
int del[maxn]; int ls[maxn],rs[maxn];
int father[maxn],siz[maxn],dep[maxn];
int idp[maxn],top[maxn],ref[maxn],hson[maxn];
void Dfs1(int now,int fa){
father[now]=fa;
dep[now]=dep[fa]+1;
siz[now]=1;
if(ls[now]){
Dfs1(ls[now],now);
siz[now]+=siz[ls[now]];
if(siz[ls[now]]>siz[hson[now]])hson[now]=ls[now];
}
if(rs[now]){
Dfs1(rs[now],now);
siz[now]+=siz[rs[now]];
if(siz[rs[now]]>siz[hson[now]])hson[now]=rs[now];
}
} int dfsclock=0;
void Dfs2(int now,int toppoint){
if(now==toppoint)nowans-=now; idp[now]=++dfsclock;
ref[dfsclock]=now;
top[now]=toppoint;
if(!hson[now])return;
Dfs2(hson[now],toppoint);
if((ls[now])&&(ls[now]!=hson[now]))Dfs2(ls[now],ls[now]);
if((rs[now])&&(rs[now]!=hson[now]))Dfs2(rs[now],rs[now]);
} struct SegmentTree{
int l,r;
int tag,mn;//siz[heavy[now]]-siz[light[son]]
int havetop;//father[top[now]]
}tree[maxn<<2];
inline void pushup(int now){
tree[now].mn=min(tree[lo].mn,tree[ro].mn);
tree[now].havetop=(tree[lo].havetop|tree[ro].havetop);
}
inline void pushdown(int now){
if(tree[now].tag){
tree[lo].tag+=tree[now].tag;
tree[ro].tag+=tree[now].tag;
tree[lo].mn+=tree[now].tag;
tree[ro].mn+=tree[now].tag;
tree[now].tag=0;
}
} void BuildTree(int now,int l,int r){
tree[now].l=l;tree[now].r=r;
tree[now].tag=0;
if(l==r){
int x=ref[l];
tree[now].mn=siz[hson[x]]-min(siz[ls[x]],siz[rs[x]]);
tree[now].havetop=0;
return;
}
int mid=(l+r)>>1;
BuildTree(lo,l,mid);
BuildTree(ro,mid+1,r);
pushup(now);
} void Updatasec(int now,int ll,int rr,int x){
if(tree[now].l>=ll&&tree[now].r<=rr){
tree[now].tag+=x;
tree[now].mn+=x;
return;
}
int mid=(tree[now].l+tree[now].r)>>1;
pushdown(now);
if(ll<=mid)Updatasec(lo,ll,rr,x);
if(rr>mid)Updatasec(ro,ll,rr,x);
pushup(now);
} void Updatatop(int now,int ll,int rr,int x){
if(ll>rr)return;
if(tree[now].r<ll)return;
if(tree[now].l>rr)return;
if(tree[now].l==tree[now].r){
tree[now].mn+=x;
return;
}
pushdown(now);
if(tree[lo].havetop)Updatatop(lo,ll,rr,x);
if(tree[ro].havetop)Updatatop(ro,ll,rr,x);
pushup(now);
} void Changetop(int now,int ll,int rr){
if(ll>rr)return;
if(tree[now].r<ll)return;
if(tree[now].l>rr)return;
if(tree[now].l==tree[now].r){
tree[now].mn=-tree[now].mn;
tree[now].havetop=1;
int x=ref[tree[now].l];
if(ls[x]==hson[x]){
nowans=nowans-ls[x]+rs[x];
hson[x]=rs[x];
}else{
nowans=nowans-rs[x]+ls[x];
hson[x]=ls[x];
}
return;
}
pushdown(now);
if(tree[lo].mn<0)Changetop(lo,ll,rr);
if(tree[ro].mn<0)Changetop(ro,ll,rr);
pushup(now);
} void Updatapoint(int now,int p,int x){
if(tree[now].l==tree[now].r){
tree[now].mn+=x;
if(tree[now].mn<0){
tree[now].mn=-tree[now].mn;
tree[now].havetop=0;
int o=ref[p];
if(ls[o]==hson[o]){
nowans=nowans-ls[o]+rs[o];
hson[o]=rs[o];
}else{
nowans=nowans-rs[o]+ls[o];
hson[o]=ls[o];
}
}
return;
}
int mid=(tree[now].l+tree[now].r)>>1;
pushdown(now);
if(p<=mid)Updatapoint(lo,p,x);
else Updatapoint(ro,p,x);
pushup(now);
} void ChangePath(int u){
int x=u;
while(x){
int tx=top[x];
Updatasec(1,idp[tx],idp[x],-1); Updatatop(1,idp[tx],idp[x]-1,2);
Changetop(1,idp[tx],idp[x]-1);
// x=tx;int fa=father[x];
// if(!fa)break;
// if(hson[fa]!=x){
// Updatapoint(1,idp[fa],2);
// }else{
// Updatapoint(1,idp[fa],0);
// }
x=father[tx];tx=top[x];
}
x=u;
while(x){
x=top[x];
int fa=father[x];
if(!fa)break;
if(hson[fa]!=x){
Updatapoint(1,idp[fa],2);
}else{
Updatapoint(1,idp[fa],0);
}
x=fa;
}
} int main(){
scanf("%d",&n);
for(int i=1;i<=n;++i){
scanf("%d%d",&ls[i],&rs[i]);
havein[ls[i]]++;
havein[rs[i]]++;
}
for(int i=1;i<=n;++i)nowans+=i;
for(int i=1;i<=n;++i)if(!havein[i])root=i; Dfs1(root,0);
Dfs2(root,root);
BuildTree(1,1,n); scanf("%d",&m);
printf("%lld\n",nowans);
del[0]=1;
while(m--){
int x;scanf("%d",&x); del[x]=1;
ChangePath(x);
if(ls[father[x]]==x){
if(del[rs[father[x]]])nowans-=x;
}else{
if(del[ls[father[x]]])nowans-=x;
}
printf("%lld\n",nowans);
}
return 0;
}
BZOJ 4855 [Jsoi2016]轻重路径的更多相关文章
- BZOJ4855 : [Jsoi2016]轻重路径
首先用树状数组维护dfs序来快速支持一个点子树大小的询问. 每次删掉一个叶子时,从根开始往叶子走,显然只有$2size[x]\leq size[father]$的点的父亲才有可能换重儿子. 从根开始往 ...
- [JSOI2016]轻重路径[树链剖分]
题意 题目链接 分析 先对原树树剖,在一次删点操作后从根节点开始二分,如果一条边从重边变成轻边,必然有 \(size_u\le \frac{1}{2}size_{rt}\) (取等号是特判对应儿子消失 ...
- BZOJ 4753 [Jsoi2016]最佳团体 | 树上背包 分数规划
BZOJ 4753 [Jsoi2016]最佳团体 | 树上背包 分数规划 又是一道卡精度卡得我头皮发麻的题-- 题面(--蜜汁改编版) YL大哥是24OI的大哥,有一天,他想要从\(N\)个候选人中选 ...
- BZOJ 2337 XOR和路径 | 高斯消元 期望 位运算
BZOJ 2337 XOR和路径 题解 这道题和游走那道题很像,但又不是完全相同. 因为异或,所以我们考虑拆位,分别考虑每一位: 设x[u]是从点u出发.到达点n时这一位异或和是1的概率. 对于所有这 ...
- 【LOJ】#2079. 「JSOI2016」轻重路径
题解 写数据结构的时候我代码就会变得非常非常长 一看别人1.5K 2.3K 我6.3K-- orzzzzz 我们很容易想到离线倒着插入,然而,有个小锅叫如果size相同保持原来的重儿子不变 我们需要写 ...
- [BZOJ 1907] 树的路径覆盖 【树形DP】
题目链接:BZOJ - 1907 题目分析 使用树形 DP,f[x][0] 表示以 x 为根的子树不能与 x 的父亲连接的最小路径数(即 x 是一个折线的拐点). f[x][1] 表示以 x 为根的子 ...
- BZOJ.3784.树上的路径(点分治 贪心 堆)
BZOJ \(Description\) 给定一棵\(n\)个点的带权树,求树上\(\frac{n\times(n-1)}{2}\)条路径中,长度最大的\(m\)条路径的长度. \(n\leq5000 ...
- bzoj 3784: 树上的路径【点分治+st表+堆】
参考:https://www.cnblogs.com/CQzhangyu/p/7071477.html 神奇的点分治序(或者叫点剖?).就是把点分治扫过的点依次放进队列里,然后发现,对于每一棵树摊到序 ...
- BZOJ 3784: 树上的路径
Description 问一棵树上前 \(k\) 大路径的边权. Sol 边分治. 非常感谢数据没有菊花图. 为了写写边分治试试然后就开了这道题. 边分治非常好想,选一条重边,分成两部分,然后分别求最 ...
随机推荐
- 003.Delphi插件之QPlugins,菜单插件加强
相比上一篇的菜单插件,这个在创建和销毁时候,做了增强,同时做了2个菜单对应的窗口 unit MenuSvc; interface uses windows, classes, SysUtils, Gr ...
- SQL计算字符串里的子字符串出现个数
在某个页面,需要显示每条记录中有几个图片文件.图片文件名列表存储在mysql表里的photo_files字段,文件名之间用一个空格分开.类似'images\rpt201503121.jpg image ...
- Flume 1.9.0 的安装(比较简单, 操作也不像老版本那么繁琐了)
之前已经完成了Hadoop集群.Hbase集群.Hive的搭建, 这次来安装一下flume-1.9.0 安装过程 将tar包上传并解压到指定目录, 并修改名称 tar -zxvf apache-flu ...
- Irecycleview 的初次使用简单介绍(irecycleview 下拉刷新上拉加载)
导包 还得加一个maven地址自己也看一下作者git把有详细解释(自己也要导入recycleview的包) 我的例子下载地址 https://www.lanzous.com/i32yzaj impl ...
- 这台计算机上缺少此项目引用的 NuGet 程序包。使用“NuGet 程序包还原”可下载这些程序包
将项目复制到其地方的时候编译会报错,按照官网方法也不行,从网上查了一个有用的方法如下 打开CSPROJ文件.删除如下代码, <Import Project="..\packages\ ...
- LeetCode1029 两地调度(贪心+java自定义排序回顾)
题目: 公司计划面试 2N 人.第 i 人飞往 A 市的费用为 costs[i][0],飞往 B 市的费用为 costs[i][1]. 返回将每个人都飞到某座城市的最低费用,要求每个城市都有 N 人抵 ...
- Git TortoiseGit github 操作
由于公司采用了分布式架构,选择的是gitlab git 来管理代码等工作,鉴于github和gitlab的相似性,网上查看好多都是用git的命令上传,或者是一部分,为此,借鉴多方网络,并进行实际操作, ...
- jenkins打包iOS 报错:error: exportArchive: The data couldn’t be read because it isn’t in the correct format.
在执行ios 打包的时候,我们通过执行下面的指令来打包ipa: mkdir arch archive_path=arch/${app_name}.xcarchive workspace_name=HP ...
- @echo off命令
在C盘下新建一个文本文档,将名字改为1.bat. 打开/编辑,输入call cmd.cmd是命令提示符.运行该文件,出现命令提示符窗口,在该窗口下可以运行各种命令.由图1.1可见,在第一行显示C:\ ...
- Listener(Web监听器、活化、钝化)
Web监听器 总共有8个 划分成三种类型 定义一个类,实现接口 注册 | 配置监听器 监听三个作用域创建和销毁 request -httpServletRequest session -httpSes ...