洛谷P3345 [ZJOI2015]幻想乡战略游戏 [动态点分治]
调了两个小时,终于过了……
凭啥人家代码80行我180行啊!!!
谁叫你大括号换行
谁叫你写缺省源
思路
显然,补给点所在的位置就是这棵树的带权重心。
考虑size已知时如何找重心:一开始设答案在\(x\),若存在\(x\)的一个子节点\(v\),使\(size_v>sum-size_v\),即\(2size_v>sum\),就往\(v\)走,重复该过程,直到走不动为止。
考虑用点分树优化这一过程:由于点分树层数不超过\(\log n\),这题又保证\(deg\leq 20\),可以暴力往下跳。
然而由于在点分树上跳,与原树有一些区别,需要搞一些奇怪的东西。
假设点分树上\(x\)点有\(v\)点为儿子且满足上述要求,在原树上有\(x\rightarrow w\)这条边,且\(w\)在\(v\)的点分树的子树里,那么就要把除了\(v\)这个连通块以外的\(size\)加到\(w\)上。
即:\(size_w+=size_x-size_v\),其中\(size\)表示点分树上的子树的\(\sum d\)。(不明白为什么要叫作\(d\),叫\(val\)不好么?)
求出重心后还要把刚才的修改操作撤销。
得到重心之后就很好做了,沿点分树往上跳就好了。
需要维护以下几样东西:
设点分树上有节点\(x\),它在点分树上的父亲是\(fa\),它在点分树上的子树为\(S\),则有:
&sumv_x=\sum_{u\in S} val_u\\
&sumV_x=\sum_{u\in S} val_u dis(u,x)\\
&sumFV_x=\sum_{u\in S} val_u dis(u,fa)
\end{align*}
\]
记录这几样东西就可以统计答案了。
听着还好,写的时候思路很容易乱,务必要理清。
不然您就会像我一样记录了一堆东西全都没用
假设\(q,n\)同阶,则最后复杂度\(O(20n\log n)\)?反正能过就行,而且还挺快。
代码
#include<bits/stdc++.h>
namespace my_std{
using namespace std;
#define pii pair<int,int>
#define fir first
#define sec second
#define MP make_pair
#define rep(i,x,y) for (int i=(x);i<=(y);i++)
#define drep(i,x,y) for (int i=(x);i>=(y);i--)
#define go(x) for (int i=head[x];i;i=edge[i].nxt)
#define sz 101010
typedef long long ll;
template<typename T>
inline void read(T& t)
{
t=0;char f=0,ch=getchar();
double d=0.1;
while(ch>'9'||ch<'0') f|=(ch=='-'),ch=getchar();
while(ch<='9'&&ch>='0') t=t*10+ch-48,ch=getchar();
if(ch=='.')
{
ch=getchar();
while(ch<='9'&&ch>='0') t+=d*(ch^48),d*=0.1,ch=getchar();
}
t=(f?-t:t);
}
template<typename T,typename... Args>
inline void read(T& t,Args&... args){read(t); read(args...);}
void file()
{
#ifndef ONLINE_JUDGE
freopen("a.txt","r",stdin);
#endif
}
inline bool chkmin(int &x,int y){return x>y?x=y,1:0;}
inline bool chkmax(int &x,int y){return x<y?x=y,1:0;}
// inline ll mul(ll a,ll b){ll d=(ll)(a*(double)b/mod+0.5);ll ret=a*b-d*mod;if (ret<0) ret+=mod;return ret;}
}
using namespace my_std;
#define v edge[i].t
namespace T
{
struct hh{int t;ll w;int nxt;}edge[sz<<1];
int head[sz],ecnt;
void make_edge(int f,int t,ll w)
{
edge[++ecnt]=(hh){t,w,head[f]};
head[f]=ecnt;
edge[++ecnt]=(hh){f,w,head[t]};
head[t]=ecnt;
}
ll val[sz];
ll Dep[sz];
void dfs(int x,int fa){ go(x) if (v!=fa) { Dep[v]=Dep[x]+edge[i].w; dfs(v,x); } }
bool vis[sz];
int size[sz],sum,rt,rtS;
void calcS(int x,int fa)
{
size[x]=1;
go(x) if (v!=fa&&!vis[v]) { calcS(v,x); size[x]+=size[v]; }
}
void findrt(int x,int fa)
{
int S=sum-size[x];
go(x) if (v!=fa&&!vis[v]) { findrt(v,x); chkmax(S,size[v]); }
if (chkmin(rtS,S)) rt=x;
}
int findrt(int x){ calcS(x,0); sum=size[x]; rtS=1e9; findrt(x,0); return rt; }
}
#define goT(x) for (int i=T::head[x];i;i=T::edge[i].nxt)
#define Tv T::edge[i].t
namespace S
{
int root;
struct hh{int t,nxt;}edge[sz<<1];
int head[sz],ecnt;
void make_edge(int f,int t){edge[++ecnt]=(hh){t,head[f]};head[f]=ecnt;}
ll sumv[sz],sumV[sz],sumFV[sz];
int ff[sz];
struct hhh{int u,uu;ll dis;};
vector<hhh>in[sz];
void calcDis(int x,int fa,int u,int uu,ll dis)
{
in[x].push_back((hhh){u,uu,dis});
goT(x) if (Tv!=fa&&!T::vis[Tv]) calcDis(Tv,x,u,uu,dis+T::edge[i].w);
}
void build(int x)
{
T::vis[x]=1;
goT(x) if (!T::vis[Tv])
{
int rt=T::findrt(Tv);
make_edge(x,rt);
calcDis(Tv,0,x,rt,T::edge[i].w);
ff[rt]=Tv;
build(rt);
}
}
void build(){root=T::findrt(1);build(root);}
void change(int x,ll s) // val_x+=s
{
T::val[x]+=s;sumv[x]+=s;
drep(t,(int)in[x].size()-1,0)
{
int u=in[x][t].u,uu=in[x][t].uu;ll D=in[x][t].dis*s;
sumv[u]+=s;sumV[u]+=D;sumFV[uu]+=D;
}
}
ll getans(int x)
{
ll ret=sumV[x];
drep(t,(int)in[x].size()-1,0)
{
int u=in[x][t].u,uu=in[x][t].uu;ll dis=in[x][t].dis;
go(u) if (v!=uu) ret+=sumFV[v]+sumv[v]*dis;
ret+=T::val[u]*dis;
}
return ret;
}
int getroot(int x)
{
go(x) if (sumv[v]*2>sumv[x])
{
int delta=sumv[x]-sumv[v];
change(ff[v],delta);
int ret=getroot(v);
change(ff[v],-delta);
return ret;
}
return x;
}
ll getans(){return getans(getroot(root));}
}
int n,Q;
int main()
{
file();
read(n,Q);
int x,y,z;
rep(i,1,n-1) read(x,y,z),T::make_edge(x,y,z);
S::build();
while (Q--)
{
read(x,y);
S::change(x,y);
printf("%lld\n",S::getans());
}
}
(压了一下行,只剩\(151\)行了)
(反正我写这么丑也没人看QwQ)
洛谷P3345 [ZJOI2015]幻想乡战略游戏 [动态点分治]的更多相关文章
- 洛谷 P3345 [ZJOI2015]幻想乡战略游戏 解题报告
P3345 [ZJOI2015]幻想乡战略游戏 题目描述 傲娇少女幽香正在玩一个非常有趣的战略类游戏,本来这个游戏的地图其实还不算太大,幽香还能管得过来,但是不知道为什么现在的网游厂商把游戏的地图越做 ...
- 洛谷P3345 [ZJOI2015]幻想乡战略游戏(动态点分治,树的重心,二分查找,Tarjan-LCA,树上差分)
洛谷题目传送门 动态点分治小白,光是因为思路不清晰就耗费了不知道多少时间去gang这题,所以还是来理理思路吧. 一个树\(T\)里面\(\sum\limits_{v\in T} D_vdist(u,v ...
- P3345 [ZJOI2015]幻想乡战略游戏 动态点分治
\(\color{#0066ff}{ 题目描述 }\) 傲娇少女幽香正在玩一个非常有趣的战略类游戏,本来这个游戏的地图其实还不算太大,幽香还能管得过来,但是不知道为什么现在的网游厂商把游戏的地图越做越 ...
- 2018.08.28 洛谷P3345 [ZJOI2015]幻想乡战略游戏(点分树)
传送门 题目就是要求维护带权重心. 因此破题的关键点自然就是带权重心的性质. 这时发现直接找带权重心是O(n)的,考虑优化方案. 发现点分树的树高是logn级别的,并且对于以u为根的树,带权重心要么就 ...
- BZOJ 3924 / Luogu P3345 [ZJOI2015]幻想乡战略游戏 (动态点分治/点分树)
题意 树的结构不变,每个点有点权,每一条边有边权,有修改点权的操作,设xxx为树中一点.求∑idist(x,i)∗a[i]\sum_idist(x,i)*a[i]i∑dist(x,i)∗a[i]的最 ...
- [ZJOI2015]幻想乡战略游戏——动态点分治
[ZJOI2015]幻想乡战略游戏 带修改下,边点都带权的重心 随着变动的过程中,一些子树内的点经过会经过一些公共边.考虑能不能对这样的子树一起统计. 把树上贡献分块. 考虑点分治算法 不妨先把题目简 ...
- 【BZOJ3924】[Zjoi2015]幻想乡战略游戏 动态树分治
[BZOJ3924][Zjoi2015]幻想乡战略游戏 Description 傲娇少女幽香正在玩一个非常有趣的战略类游戏,本来这个游戏的地图其实还不算太大,幽香还能管得过来,但是不知道为什么现在的网 ...
- [BZOJ3924][ZJOI2015]幻想乡战略游戏(动态点分治)
题目描述 傲娇少女幽香正在玩一个非常有趣的战略类游戏,本来这个游戏的地图其实还不算太大,幽香还能管得过来,但是不知道为什么现在的网游厂商把游戏的地图越做越大,以至于幽香一眼根本看不过来,更别说和别人打 ...
- 【bzoj3924】[Zjoi2015]幻想乡战略游戏 动态点分治
题目描述 傲娇少女幽香正在玩一个非常有趣的战略类游戏,本来这个游戏的地图其实还不算太大,幽香还能管得过来,但是不知道为什么现在的网游厂商把游戏的地图越做越大,以至于幽香一眼根本看不过来,更别说和别人打 ...
随机推荐
- 【Linux】排序命令sort
很多时候,我们都会去计算一次数据里头的相同型态的数据总数,举例来说, 使用 last 可以查得这个月份有登陆主机者的身份.那么我可以针对每个使用者查出他们的总登陆次数吗? 此时就得要排序与计算之类的命 ...
- 【poj1741】Tree 树的点分治
题目描述 Give a tree with n vertices,each edge has a length(positive integer less than 1001). Define dis ...
- 复习java web之jsp入门_El表达式_JSTL标签库
JSP 技术掌握:JSP语法 + EL + JSTL 为什么sun推出 JSP技术 ? Servlet 生成网页比较复杂,本身不支持HTML语法,html代码需要通过response输出流输出,JSP ...
- luogu P3241 [HNOI2015]开店
传送门 (下面记年龄为\(a_x\))题目要求的是\[\sum_{x=1}^{n} [a_x\in [l,r]]*dis(x,u)=\sum_{x=1}^{n} [a_x\in [l,r]]*de_x ...
- GDI+学习---1.初识GDI+
---恢复内容开始--- GDI+: GDI+由一组C++类实现,是对于GDI的继承,GDI+不仅优化了大部分GDI性能而且提供了更多特性.允许应用程序开发者将信息显示在显示器或者打印机上,而无需考虑 ...
- JAVA知识笔记
1.对于volatile修饰的变量,jvm虚拟机只是保证从主内存加载到线程工作内存的值是最新的.更详细介绍参考http://blog.csdn.net/libing13820393394/articl ...
- Nginx系列1:ubuntu16.04编译出适合自己的nginx服务器
1.下载nginx nginx官网:nginx.org tar.gz文件 解压缩命令: wget https://nginx.org/download/nginx-1.14.2.tar.gz #下载n ...
- openstack-----各种系统镜像制作
本章内容 1.centos镜像制作 2.windows镜像制作 3.ubunt镜像制作 一.centos7镜像制作 1.检查系统是否支持kvm: egrep "(vmx|svm)&q ...
- zookeeper的ACL权限控制
ACL:Access Control List 访问控制列表 1. 简介 0.概述 ACL 权限控制,使用:scheme:id:perm 来标识,主要涵盖 3 个方面: 权限模式(Scheme): ...
- 使用WinIo32绕过密码控件实现自动登录
通过winIO32绕过密码控件,实现自动登录 环境: vmware上安装windows 32位系统:windows xp / windows 7 selenium版本: 3.11.0 IEDriver ...