BZOJ3435 & 洛谷3920 & UOJ55:[WC2014]紫荆花之恋
https://www.lydsy.com/JudgeOnline/problem.php?id=3435
https://www.luogu.org/problemnew/show/P3920
强强和萌萌是一对好朋友。有一天他们在外面闲逛,突然看到前方有一棵紫荆树。这已经是紫荆花飞舞的季节了,无数的花瓣以肉眼可见的速度从紫荆树上长了出来。仔细看看的话,这个大树实际上是一个带权树。每个时刻它会长出一个新的叶子节点。每个节点上有一个可爱的小精灵,新长出的节点上也会同时出现一个新的小精灵。小精灵是很萌但是也很脆弱的生物,每个小精灵 i 都有一个感受能力值Ri ,小精灵 i, j 成为朋友当且仅当在树上 i 和 j 的距离 dist(i,j) ≤ Ri + R! ,其中 dist(i, j)表示在这个树上从 i 到 j 的唯一路径上所有边的边权和。强强和萌萌很好奇每次新长出一个叶子节点之后,这个树上总共有几对朋友。
我们假定这个树一开始为空,节点按照加入的顺序从 1开始编号。由于强强非常好奇, 你必须在他每次出现新节点后马上给出总共的朋友对数,不能拖延哦。
对着题解的指针魔改了一个在洛谷开O2可过,其他网站可过的数组版。
题解请见:https://www.luogu.org/blog/user15268/solution-p3920
其实理解起来还是很好理解的,关键就在于你的卡常姿势以及代码能力的问题了。
(然而考场谁写谁跪的谁敢写这东西……)
简单叙述题解:
树不会动的时候显然点分治,把条件拆开即可用平衡树维护做(比较基础的点分治,具体做法看参考)。
树会动思考点分治并非一定要重心,只有当差的太离谱时为了效率将点重新变为重心就行。
这就是替罪羊树的思想了:暴力维护,失衡时暴力重建。我们把这样的点所构成的结构叫做点分树。
那最开始的点分树就随便搞,失衡了再静态维护一下即可。
总代码压行后198行……应该不影响阅读,因为linux的问题缩进被吃了一些请见谅。
#include<cmath>
#include<stack>
#include<vector>
#include<cstdio>
#include<cctype>
#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std;
typedef long long ll;
typedef double dl;
const int N=1e5+;
const int mod=1e9;
const dl alpha=0.812345;
inline int read(){
int X=,w=;char ch=;
while(!isdigit(ch)){w|=ch=='-';ch=getchar();}
while(isdigit(ch))X=(X<<)+(X<<)+(ch^),ch=getchar();
return w?-X:X;
}
inline void put(ll x){
if(x>)put(x/);
putchar(x%+);
}
struct node{
int to,nxt,w;
}e[N*];
int cnt,head[N],r[N],fa[N],size[N],son[N],dis[N],n,q[N];
vector<int>anc[N],id[N],sons[N];
bool vis[N];
ll ans;
int t[*N][],s[*N],p[*N],w[*N];
int sz,rt[*N];
stack<int>bin;
inline void add(int u,int v,int w){
e[++cnt].to=v;e[cnt].w=w;e[cnt].nxt=head[u];head[u]=cnt;
}
inline int rand(){
static int seed=;
return seed=(ll)seed*%;
}
inline int f(int x){return x+n;}
inline int getnod(){
if(!bin.empty()){
int u=bin.top();bin.pop();
return u;
}
return ++sz;
}
inline void upt(int k){
s[k]=s[t[k][]]+s[t[k][]]+;
}
inline void zig(int &k){
int y=t[k][];t[k][]=t[y][];t[y][]=k;
s[y]=s[k];upt(k);
k=y;
}
inline void zag(int &k){
int y=t[k][];t[k][]=t[y][];t[y][]=k;
s[y]=s[k];upt(k);
k=y;
}
inline void del(int &k){
if(!k)return;
bin.push(k);
if(t[k][])del(t[k][]);
if(t[k][])del(t[k][]);
k=;
}
inline void insert(int &k,int val){
if(!k){
k=getnod();w[k]=val;p[k]=rand();
s[k]=;t[k][]=t[k][]=;
return;
}
else ++s[k];
if(val<=w[k]){
insert(t[k][],val);
if(p[t[k][]]<p[k])zig(k);
}else{
insert(t[k][],val);
if(p[t[k][]]<p[k])zag(k);
}
}
inline int find(int x,int val){
int res=;
while(x){
if(val<w[x])x=t[x][];
else res+=s[t[x][]]+,x=t[x][];
}
return res;
}
int calcg(int st){
int R=,g,maxn=n;
q[++R]=st;fa[st]=;
for(int L=;L<=R;++L){
int u=q[L];
size[u]=;son[u]=;
for(int i=head[u];i;i=e[i].nxt){
int v=e[i].to;
if(!vis[v]||v==fa[u])continue;
fa[v]=u;q[++R]=v;
}
}
for(int L=R;L>=;--L){
int u=q[L],v=fa[u];
if(R-size[u]>son[u])son[u]=R-size[u];
if(son[u]<maxn)g=u,maxn=son[u];
if(!v)break;
size[v]+=size[u];
if(size[u]>son[v])son[v]=size[u];
}
return g;
}
inline void dac(int st,int par){
int R=,g=calcg(st);vis[g]=;
q[++R]=g;fa[g]=;dis[g]=;
for(int L=;L<=R;++L){
int u=q[L];
for(int j=head[u];j;j=e[j].nxt){
int v=e[j].to;
if(!vis[v]||v==fa[u])continue;
fa[v]=u;dis[v]=dis[u]+e[j].w;
q[++R]=v;
}
}
for(int L=;L<=R;++L){
int u=q[L];
id[u].push_back(g);
anc[u].push_back(dis[u]);
sons[g].push_back(u);
insert(rt[g],dis[u]-r[u]);
if(par)insert(rt[f(g)],anc[u][anc[u].size()-]-r[u]);
}
for(int i=head[g];i;i=e[i].nxt){
int v=e[i].to;
if(vis[v])dac(v,g);
}
}
vector<int>tmp;
inline void rebuild(int u,int par){
tmp=sons[u];
int len=anc[par].size();
for(int i=;i<tmp.size();++i){
int v=tmp[i];vis[v]=;
sons[v].clear();
anc[v].resize(len);
id[v].resize(len);
del(rt[v]);del(rt[f(v)]);
}
dac(u,par);
}
inline void check(int u){
for(int i=;i<id[u].size();++i){
insert(rt[id[u][i]],anc[u][i]-r[u]);
if(i)insert(rt[f(id[u][i])],anc[u][i-]-r[u]);
}
for(int i=;i<id[u].size()-;++i){
int sz_f=s[rt[id[u][i]]];
int sz_s=s[rt[id[u][i+]]];
if(sz_f<=)return;
if(sz_s>alpha*sz_f){
rebuild(id[u][i],(!i)?:id[u][i-]);
return;
}
}
}
inline int solve(int u,int v,int w){
int res=;
anc[u]=anc[v];id[u]=id[v];
anc[u].push_back(-w);id[u].push_back(u);
for(int i=;i<anc[u].size();++i){
anc[u][i]+=w;
sons[id[u][i]].push_back(u);
res+=find(rt[id[u][i]],r[u]-anc[u][i]);
if(i)res-=find(rt[f(id[u][i])],r[u]-anc[u][i-]);
}
return res;
}
int main(){
read();n=read();
for(int i=;i<=n;++i){
int u=read()^(ans%mod),w=read();r[i]=read();
if(i==){
anc[i].push_back();
id[i].push_back(i);
sons[i].push_back(i);
insert(rt[i],-r[i]);
puts("");
continue;
}
add(u,i,w);add(i,u,w);
ans+=solve(i,u,w);
check(i);
put(ans);putchar('\n');
}
return ;
}
BZOJ3435 & 洛谷3920 & UOJ55:[WC2014]紫荆花之恋的更多相关文章
- 【BZOJ3435】[Wc2014]紫荆花之恋 替罪点分树+SBT
[BZOJ3435][Wc2014]紫荆花之恋 Description 强强和萌萌是一对好朋友.有一天他们在外面闲逛,突然看到前方有一棵紫荆树.这已经是紫荆花飞舞的季节了,无数的花瓣以肉眼可见的速度从 ...
- bzoj 3435: [Wc2014]紫荆花之恋 替罪羊树维护点分治 && AC400
3435: [Wc2014]紫荆花之恋 Time Limit: 240 Sec Memory Limit: 512 MBSubmit: 159 Solved: 40[Submit][Status] ...
- BZOJ 3435: [Wc2014]紫荆花之恋
二次联通门 : BZOJ 3435: [Wc2014]紫荆花之恋 二次联通门 : luogu P3920 [WC2014]紫荆花之恋 /* luogu P3920 [WC2014]紫荆花之恋 怀疑人生 ...
- luogu P3920 [WC2014]紫荆花之恋
LINK:紫荆花之恋 每次动态加入一个节点 统计 有多少个节点和当前节点的距离小于他们的权值和. 显然我们不能n^2暴力. 考虑一个简化版的问题 树已经给出 每次求某个节点和其他节点的贡献. 不难想到 ...
- BZOJ3435[Wc2014]紫荆花之恋——动态点分治(替罪羊式点分树套替罪羊树)
题目描述 强强和萌萌是一对好朋友.有一天他们在外面闲逛,突然看到前方有一棵紫荆树.这已经是紫荆花飞舞的季节了,无数的花瓣以肉眼可见的速度从紫荆树上长了出来.仔细看看的话,这个大树实际上是一个带权树.每 ...
- 【bzoj3435】[Wc2014]紫荆花之恋 替罪点分树套SBT
题目描述 强强和萌萌是一对好朋友.有一天他们在外面闲逛,突然看到前方有一棵紫荆树.这已经是紫荆花飞舞的季节了,无数的花瓣以肉眼可见的速度从紫荆树上长了出来.仔细看看的话,这个大树实际上是一个带权树.每 ...
- BZOJ3435: [Wc2014]紫荆花之恋(替罪羊树,Treap)
Description 强强和萌萌是一对好朋友.有一天他们在外面闲逛,突然看到前方有一棵紫荆树.这已经是紫荆花飞舞的季节了,无数的花瓣以肉眼可见的速度从紫荆树上长了出来.仔细看看的话,这个大树实际上是 ...
- bzoj3435 [Wc2014]紫荆花之恋
如果这棵树不变的话,就是一个裸的点分树套平衡树,式子也很好推$di+dj<=ri+rj$,$ri-di>=dj-rj$ 平衡树维护$dj-rj$,然后查$ri-di$的$rank$即可. ...
- bzoj3435 [Wc2014]紫荆花之恋(动态点分治+替罪羊树)
传送门(权限) 传送门(非权限) 题解 我终终终终终终于做出来啦!!! 作为一个没有学过替罪羊树的蒟蒻现场学了一下替罪羊树,作为一个平衡树都写数组版本的看着大佬的指针题解无语只能硬去理解然后照着抄了一 ...
随机推荐
- MySQL高级-全局查询日志
注意:全局查询日志不要在生成环境中启用 一.配置启用 二.编码启用
- 利用Powershell查询AD中账号属性
标签:AD账号信息 最后登录时间 最后修改密码.SID 账号SID 原创作品,允许转载,转载时请务必以超链接形式标明文章 原始出处 .作者信息和本声明.否则将追究法律责任.http://hubuxcg ...
- CentOS 使用PostFix搭建邮件服务器
搭建环境: 关于PostFix是什么以及邮件服务器接受发送邮件流程网上有很多文章,这里就不再写了,这里只记录如何搭建邮件服务器,使用PostFix接受发送邮件 CentOS6.8 32位,postfi ...
- Python3安装pywin32模块
假如你安装的是Python3.6, 那么可以直接用PyCharm或者pip安装pywin32模块: 但是, 由于我安装的是Python3.7, 所以PyCharm或者pip都无法成功安装pywin32 ...
- 【WXS数据类型】Number
Number包括整数与小数. 属性: 名称 返回 说明 [Number].constructor 值为字符串“Number” 返回该类型的结构字符串 方法: 原型:[Number].toString( ...
- Java学习 · 初识 面向对象深入二
面向对象深入 1. 抽象类 a) 声明 i. 抽象方法和抽象类必须用abstract来修饰 ii. 没有方法体,不需要实现 b) ...
- maven 安装、配置
简介: maven 大大提高项目开发速度 编译---打包---测试--安装 一条龙 maven将项目构建的过程标准化,每一个阶段使用一个命令完成,下面是构建过程一些阶段 清理 mvn cl ...
- springMVC第二章
springMVC第二章 一.URL 映射 可以同时设置多个URL来访问某个控制器或方法.设置value属性: @RequestMapping(value= {"/grade",& ...
- https的主体过程
https其实就是基于SSL的http.加密后的http信息按理是不会被篡改和查看的. https的过程总体上是按照下面来进行的: 1.客户端发起请求,服务端返回一个SSL证书,证书里面有一公钥A. ...
- NMAP-主机扫描
1.全面扫描 2.扫描指定段 3.ping扫描 只进行ping操作,十分隐蔽 4.无ping扫描 适用于防火墙禁止ping 5.TCP SYN扫描 6.TCP ACK扫描 7.UDP扫描 8.ICMP ...