[ZJOI2015]幻想乡战略游戏——动态点分治
带修改下,边点都带权的重心
随着变动的过程中,一些子树内的点经过会经过一些公共边。考虑能不能对这样的子树一起统计。
把树上贡献分块。
考虑点分治算法
不妨先把题目简化一下:
假设没有修改,多次询问,每次给定一个s,求$\sum d_v*dis(s,v)$
为了让一块可以一起统计,
我们设:
$sum[x]:$x为根的点分治树管辖部分的权值和
$df[x]$:x为根的点分治树管辖部分到x分治树father的带权距离和$\sum d[v]*dis[v,fa[x]]$
$dm[x]:$ x为根的点分治树管辖部分到x自己的带权距离和
自底向上统计总距离
开始加上$dm[s]$,然后不断加上$dis(s,fa)*(sum[fa]-sum[las])+dm[fa]-df[las]$
las代表上一个father
$dis(s,fa)$可以开始点分治的时候保存(反正就log个)
对于一个s,可以$logn$找到答案
$dm,sum,df$也可以轻而易举$logn$修改
问题是现在我们不知道s是哪一个。
我们比较熟悉一个叫货仓选址问题。
这个重心,很类似于带权中点。
从分治树的root开始,枚举原树上的出边,到y,y所在的这一层的分治树的father叫做son。如果存在一个子树的sum[son]要大于x的剩下分支的点值总和,那么重心一定在son里。
(简单证明:
如果存在sum[son]>sum[x]-sum[son],那么这样的son一定只有一个
存在往son走,减去的代价多,增加的代价少。存在代价更小的情况。(至少是y)
而往其他子树走,代价一定是单调递增上涨的。不可能更优。
所以,重心一定在son管辖的分治树这块区域
)
可以递归处理下去。
但是一个问题是,

如果到了son位置,但是对于son的1号子树,权值和必须考虑上B的所有权值和。
否则肯定不能直接走。
我们还要返回去考虑father们的权值?
发现难以处理。因为之后的划分可能较多。还要$logn$暴力找father
发现B的权值只有在询问包含y的子树的块才会用到。
所以,我们干脆直接把y点的权值加上sum[B]。回溯回来再减去。
然后就可以放心大胆查询子树的权值和了。
(因为这个小trick瞎写了半天。。。。)
由于只会找到一个son,所以,每次找s的复杂度是:$O(20logn+log^2n)$
就可以了。
(虽然暴力$20*log^2n$每次也可以过,因为时限6s)
代码:
// luogu-judger-enable-o2
#include<bits/stdc++.h>
#define reg register int
#define il inline
#define numb (ch^'0')
using namespace std;
typedef long long ll;
il void rd(int &x){
char ch;x=;bool fl=false;
while(!isdigit(ch=getchar()))(ch=='-')&&(fl=true);
for(x=numb;isdigit(ch=getchar());x=x*+numb);
(fl==true)&&(x=-x);
}
namespace Miracle{
const int N=1e5+;
int n,m;
struct node{
int nxt,to;
int son;
ll val;
}e[*N];
int hd[N],cnt;
void add(int x,int y,ll z){
e[++cnt].nxt=hd[x];
e[cnt].to=y;
e[cnt].val=z;
hd[x]=cnt;
}
ll has[N];
int rt;
ll dm[N],df[N],sum[N];
ll dis[N][];
int fa[N];
int sz[N],mxsz[N];
int nowsz;
int gen;
int dep[N];
bool vis[N];
void dfs1(int x,int ff,int d){
dep[x]=d;
sz[x]=;mxsz[x]=;
for(reg i=hd[x];i;i=e[i].nxt){
int y=e[i].to;
if(y==ff) continue;
if(vis[y]) continue;
dfs1(y,x,d);
sz[x]+=sz[y];
mxsz[x]=max(mxsz[x],sz[y]);
}
mxsz[x]=max(mxsz[x],nowsz-sz[x]);
if(mxsz[x]<=nowsz/){
rt=x;
}
}
void dfs2(int x,int ff,int d){
sz[x]=;
for(reg i=hd[x];i;i=e[i].nxt){
int y=e[i].to;
if(y==ff||vis[y]) continue;
dis[y][d]=dis[x][d]+e[i].val;
dfs2(y,x,d);
sz[x]+=sz[y];
}
}
int divi(int x,int ff,int d){
rt=;
dfs1(x,,d);
dis[rt][d]=;
fa[rt]=ff;
dfs2(rt,,d); vis[rt]=;
int now=rt;
for(reg i=hd[now];i;i=e[i].nxt){
int y=e[i].to;
if(vis[y]) continue;
nowsz=sz[y];
e[i].son=divi(y,now,d+);
}
return now;
}
void upda(int x,ll c){//c is change
int gg=x;
int nd=dep[x];
while(x){
sum[x]+=c;
df[x]+=c*dis[gg][nd-];
dm[x]+=c*dis[gg][nd]; x=fa[x];
--nd;
}
}
int query(int x,int d){
//cout<<" querying "<<x<<" || "<<" "<<d<<endl;
for(reg i=hd[x];i;i=e[i].nxt){
int y=e[i].to;
if(dep[y]<dep[x]) continue;
int lp=e[i].son;
if(sum[lp]>sum[x]-sum[lp]){
upda(y,sum[x]-sum[lp]);
int ret=query(lp,d+);
upda(y,sum[lp]-sum[x]);
return ret;
}
}
return x;
}
ll calc(int x){
ll ret=dm[x];
int gg=x;
int las=x;
x=fa[x];
int nd=dep[x];
while(x){
ret+=(dm[x]-df[las])+(sum[x]-sum[las])*dis[gg][nd];
--nd;
las=x;
x=fa[x];
}
return ret;
}
int main(){
rd(n);rd(m);
int x,y;
ll z;
for(reg i=;i<=n-;++i){
rd(x);rd(y);scanf("%lld",&z);
add(x,y,z);add(y,x,z);
}
nowsz=n;
gen=divi(,,);
//cout<<" ffafaf "<<endl;
// for(reg i=1;i<=n;++i){
// cout<<i<<" : "<<fa[i]<<endl;
// }
//cout<<" gen "<<gen<<endl;
while(m--){
rd(x);scanf("%lld",&z);
upda(x,z);
has[x]+=z;//warning!!!!
// for(reg i=1;i<=n;++i){
// cout<<i<<" : "<<sum[i]<<" "<<dm[i]<<" "<<df[i]<<endl;
// }
int core=query(gen,);
printf("%lld\n",calc(core)); }
return ;
} }
int main(){
// freopen("data.in","r",stdin);
// freopen("my.out","w",stdout);
Miracle::main();
return ;
} /*
Author: *Miracle*
Date: 2018/11/29 8:55:17
*/
[ZJOI2015]幻想乡战略游戏——动态点分治的更多相关文章
- 【BZOJ3924】[Zjoi2015]幻想乡战略游戏 动态树分治
[BZOJ3924][Zjoi2015]幻想乡战略游戏 Description 傲娇少女幽香正在玩一个非常有趣的战略类游戏,本来这个游戏的地图其实还不算太大,幽香还能管得过来,但是不知道为什么现在的网 ...
- [BZOJ3924][ZJOI2015]幻想乡战略游戏(动态点分治)
题目描述 傲娇少女幽香正在玩一个非常有趣的战略类游戏,本来这个游戏的地图其实还不算太大,幽香还能管得过来,但是不知道为什么现在的网游厂商把游戏的地图越做越大,以至于幽香一眼根本看不过来,更别说和别人打 ...
- P3345 [ZJOI2015]幻想乡战略游戏 动态点分治
\(\color{#0066ff}{ 题目描述 }\) 傲娇少女幽香正在玩一个非常有趣的战略类游戏,本来这个游戏的地图其实还不算太大,幽香还能管得过来,但是不知道为什么现在的网游厂商把游戏的地图越做越 ...
- 【bzoj3924】[Zjoi2015]幻想乡战略游戏 动态点分治
题目描述 傲娇少女幽香正在玩一个非常有趣的战略类游戏,本来这个游戏的地图其实还不算太大,幽香还能管得过来,但是不知道为什么现在的网游厂商把游戏的地图越做越大,以至于幽香一眼根本看不过来,更别说和别人打 ...
- ZJOI2015 幻想乡战略游戏 动态点分治_树链剖分_未调完
Description 傲娇少女幽香正在玩一个非常有趣的战略类游戏,本来这个游戏的地图其实还不算太大,幽香还能管得过来,但是不知道为什么现在的网游厂商把游戏的地图越做越大,以至于幽香一眼根本看不过来, ...
- BZOJ 3924: [Zjoi2015]幻想乡战略游戏(动态点分治)
这种动态点分治嘛,GDKOI时听打到了,也有同学讲到了,所以印象比较深刻也就想出来了,然后就在实现方面卡了好久= = 不得不说CLJ说得真的太简单了,实现方面根本没提. 首先我们可以先用树分治构建出这 ...
- BZOJ 3924 / Luogu P3345 [ZJOI2015]幻想乡战略游戏 (动态点分治/点分树)
题意 树的结构不变,每个点有点权,每一条边有边权,有修改点权的操作,设xxx为树中一点.求∑idist(x,i)∗a[i]\sum_idist(x,i)*a[i]i∑dist(x,i)∗a[i]的最 ...
- 洛谷P3345 [ZJOI2015]幻想乡战略游戏 [动态点分治]
传送门 调了两个小时,终于过了-- 凭啥人家代码80行我180行啊!!! 谁叫你大括号换行 谁叫你写缺省源 思路 显然,补给点所在的位置就是这棵树的带权重心. 考虑size已知时如何找重心:一开始设答 ...
- bzoj 3924 [Zjoi2015]幻想乡战略游戏——动态点分治(暴力移动找重心)
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3924 度数只有20,所以从一个点暴力枚举其出边,就能知道往哪个方向走. 知道方向之后直接走到 ...
随机推荐
- Appium+python 自动发送邮件(1)(转)
(原文:https://www.cnblogs.com/fancy0158/p/10056091.html) SMTP:简单传输协议,实在Internet上传输Email的事实标准. Python的s ...
- MySQL☞左外链接与右外连接
外链接查询:即要查询有关联关系的数据,还要查询没有关联关系的数据.(个人理解为:表A和表B两个关联的列中)如下图: emmm,简单的来说两个表的关联关系:book.bid=bookshop.id,他们 ...
- Phaser3游戏三角学应用--一只跟随屏幕点击位置游动的鱼
fish fish 资源图: fish-136x80.png undersea-bg.png 代码 var config = { type: Phaser.AUTO, parent: 'iFiero' ...
- MAC下Android的Eclipse开发环境搭建
原文链接:https://www.cnblogs.com/macro-cheng/archive/2011/09/30/android-001.html 一.Eclipse的下载 到网站:http:/ ...
- Java基础知识:Java实现Map集合二级联动4
comboBox.setModel(new DefaultComboBoxModel(getProvince())); // 添加省份信息 final JLabel label = new JLabe ...
- windows更改MySQL存储路径
在C:\ProgramData\MySQL\MySQL Server 5.7文件夹 my.ini是默认的配置文件.在这里我们只更改数据存储路径.不更改配置文件 1 # Path to the data ...
- Linux系统网络安装——基于pxe+dhcp+nfs+tftp+kickstart
原文发表于:2010-09-05 转载至cu于:2012-07-21 一.原理简介 PXE(preboot execute environment)工作于Client/Server的网络模式,支持工作 ...
- 小数第n位:高精度
小数第n位 问题描述 我们知道,整数做除法时,有时得到有限小数,有时得到无限循环小数. 如果我们把有限小数的末尾加上无限多个0,它们就有了统一的形式. 本题的任务是:在上面的约定下,求整数除法小数点后 ...
- java读取excel或者csv时日期格式数据处理
背景:最近写一个通过excel批量导入数据的功能,里面含有时间,但是java读取之后把时间转为了距离1990年1月1号的天数,比如excel中时间为2018/9/16 18:30,java读取之后变成 ...
- node项目设置环境变量
在UNIX系统中: $ NODE_ENV=production node app 在Windows中: $ set NODE_ENV=production $ node app 这些环境变量会出现在程 ...