[UOJ UR #4追击圣诞老人]
来自FallDream的博客,未经允许,请勿转载, 谢谢。
考虑直接维护一个堆,然后往里面丢链,并且取出k个堆顶就行了。
然后就需要分类讨论啥的,给你的三个点变成两条链,每次取出一条链之后选择权值最小的再劈成两条链丢进去。
卡空间 所以树剖,不选择倍增
复杂度O((n+k)logn)
#include<iostream>
#include<cstdio>
#include<queue>
#define MN 500000
#define N 524288
#define ll long long
using namespace std;
inline int read()
{
int x=,f=;char ch=getchar();
while(ch<''||ch>''){if(ch=='-')f=-;ch=getchar();}
while(ch>=''&&ch<=''){x=x*+ch-'';ch=getchar();}
return x*f;
}
struct data{int x,y;ll X;
friend bool operator <(const data&x,const data&y){return x.X>y.X;}
data operator + (ll y)
{
data c=*this;c.X+=y;
return c;
}
};
priority_queue<data> q;
int n,k,w[MN+],dep[MN+],head[MN+],cnt=,top[MN+],mx[MN+];
int s[MN+],p[MN+],fa[MN+],T[N*+],dfn[MN+],dn=;
struct edge{int to,next;}e[MN+];
inline void ins(int f,int t){e[++cnt]=(edge){t,head[f]};head[f]=cnt;}
vector<data>v[MN+];
int Merge(int x,int y){return w[x]>w[y]?y:x;}
int query(int l,int r)
{
int sum=;
for(l+=N-,r+=N+;l^r^;l>>=,r>>=)
{
if(~l&) sum=Merge(sum,T[l+]);
if( r&) sum=Merge(sum,T[r-]);
}
return sum;
}
inline int Up(int x,int k)
{
int z=dep[x]-k;
for(;dep[top[x]]>z;x=fa[top[x]]);
return p[dfn[top[x]]+z-dep[top[x]]];
}
int lca(int x,int y)
{
for(;top[x]!=top[y];x=fa[top[x]])
if(dep[top[x]]<dep[top[y]]) swap(x,y);
return dep[x]<dep[y]?x:y;
} pair<int,int> Query(int x,int y)
{
int res=;
for(;top[x]!=top[y];x=fa[top[x]])
{
if(dep[top[x]]<dep[top[y]]) swap(x,y);
res=Merge(res,s[x]);
}
if(dfn[x]>dfn[y]) swap(x,y);
res=Merge(res,query(dfn[x],dfn[y]));
return make_pair(x,res);
} data Insert(int x,int y,ll v)
{
// cout<<"Insert"<<x<<" "<<y<<" "<<v<<endl;
pair<int,int> z=Query(x,y);
// cout<<"CalcOK"<<z.first<<" "<<z.second<<endl;
return (data){x,y,w[z.second]}+v;
} void Solve(int x,int y,int z,int l,ll Add)
{
// cout<<"Solve"<<x<<" "<<y<<" "<<z<<" "<<l<<" "<<Add<<endl;
if(z==l)
{
if(x!=z) q.push(Insert(x,Up(x,dep[x]-dep[z]-),Add));
if(y!=z) q.push(Insert(y,Up(y,dep[y]-dep[z]-),Add));
return;
}
if(!(dep[x]>=dep[z]&&Up(x,dep[x]-dep[z])==z)) swap(x,y);
if(x!=z) q.push(Insert(x,Up(x,dep[x]-dep[z]-),Add));
q.push(Insert(fa[z],y,Add));
} void Dfs(int x,int tp)
{
top[x]=tp;p[dfn[x]=++dn]=x;
if(tp==x) s[x]=x; else s[x]=Merge(s[fa[x]],x);
if(mx[x]) Dfs(mx[x],tp);
for(int i=head[x];i;i=e[i].next)
if(e[i].to!=mx[x]) Dfs(e[i].to,e[i].to);
} void Pre(int x)
{
top[x]=;mx[x]=;
for(int i=head[x];i;i=e[i].next)
{
Pre(e[i].to);
top[x]+=top[e[i].to];
if(top[e[i].to]>top[mx[x]]) mx[x]=e[i].to;
}
} int main()
{
n=read();k=read();w[]=1e9;
for(int i=;i<=n;++i) w[i]=read(),q.push((data){i,i,w[i]});
for(int i=;i<=n;++i) ins(fa[i]=read(),i),dep[i]=dep[fa[i]]+;
Pre();Dfs(,);
for(int i=;i<=n;++i) T[i+N]=p[i];
for(int i=N;i;--i) T[i]=Merge(T[i<<],T[i<<|]);
for(int i=;i<=n;++i)
{
int x=read(),y=read(),z=read();
if(x==y&&y==z){v[i].push_back((data){x,x,w[x]});continue;}
if(x==y) swap(x,z);v[i].push_back(Insert(x,y,));
if(z==y||z==x) continue;
int l1=lca(x,z),l2=lca(y,z),L=lca(x,y);
if(dep[z]<dep[L]) v[i].push_back(Insert(fa[L],z,));
else if(z!=l1&&z!=l2)
{
if(dep[l1]<dep[l2]) swap(l1,l2);
int Z=Up(z,dep[z]-dep[l1]-);
v[i].push_back(Insert(z,Z,));
}
}
for(int i=;i<=k;++i)
{
data x=q.top();q.pop();printf("%lld\n",x.X);int z=Query(x.x,x.y).second;
//printf("%d %d %d %d %lld\n",x.x,x.y,x.l,x.z,x.X);
Solve(x.x,x.y,z,lca(x.x,x.y),x.X-w[z]);
for(int j=;j<v[z].size();++j) q.push(v[z][j]+x.X);
}
return ;
}
[UOJ UR #4追击圣诞老人]的更多相关文章
- UOJ#53. 【UR #4】追击圣诞老人 树链剖分 k短路
原文链接https://www.cnblogs.com/zhouzhendong/p/UOJ53.html 题意 给定一棵有 n 个节点的树. 每一个点有一个权值. 对于每一个 $i$ 给定三个参数 ...
- [UOJ UR #2]树上GCD
来自FallDream的博客,未经允许,请勿转载,谢谢. 传送门 看完题目,一般人都能想到 容斥稳了 .这样我们只要统计有多少点对满足gcd是i的倍数. 考虑长链剖分,每次合并的时候,假设我已经求出轻 ...
- [UOJ UR#16]破坏发射台
来自FallDream的博客,未经允许,请勿转载,谢谢. 传送门 先考虑n是奇数的情况,很容易想到一个dp,f[i][0/1]表示转移到第i个数,第i个数是不是第一个数的方案数,然后用矩阵乘法优化一下 ...
- UOJ UR#9 App管理器
题目传送门 题目大意大概就是给你一个混合图(既有有向边又有无向边),对于每条无向边,u-v,问删去u->v,或删去v->u那条可以使新图强连通.(保证数据有解). 这道题前几个数据点送分. ...
- 【UOJ#33】【UR#2】树上GCD 有根树点分治 + 容斥原理 + 分块
#33. [UR #2]树上GCD 有一棵$n$个结点的有根树$T$.结点编号为$1…n$,其中根结点为$1$. 树上每条边的长度为$1$.我们用$d(x,y)$表示结点$x,y$在树上的距离,$LC ...
- uoj #118. 【UR #8】赴京赶考 水题
#118. [UR #8]赴京赶考 Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://uoj.ac/problem/118 Description ...
- uoj #31. 【UR #2】猪猪侠再战括号序列 贪心
#31. [UR #2]猪猪侠再战括号序列 Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://uoj.ac/problem/31 Descript ...
- UOJ 241. 【UR #16】破坏发射台 [矩阵乘法]
UOJ 241. [UR #16]破坏发射台 题意:长度为 n 的环,每个点染色,有 m 种颜色,要求相邻相对不能同色,求方案数.(定义两个点相对为去掉这两个点后环能被分成相同大小的两段) 只想到一个 ...
- 【UOJ#51】【UR #4】元旦三侠的游戏(博弈论)
[UOJ#51][UR #4]元旦三侠的游戏(博弈论) 题面 UOJ 题解 考虑暴力,\(sg[a][b]\)记录\(sg\)函数值,显然可以从\(sg[a+1][b]\)和\(sg[a][b+1]\ ...
随机推荐
- css精简命名
想写写前言啥的,发现自己是前言无能星人. 简单吐吐槽好了,来到新公司,接手公司之前的项目,我想着也就是改改bug,慢慢来吧,粗略看了看这个项目的代码,目前仅看了html和css样式的,忍不住吐血三升. ...
- 第二篇:Python数据类型
一.引子 1.什么是数据? x= #是我们要存储的数据 2.为何数据要分不同的类型 数据是用来表示状态的,不同的状态就应该用不同的类型的数据去表示 3.数据类型 数字(整型,长整型,浮点型,复数) 字 ...
- Python之旅.第四章.模块与包 4.02
一.模块的使用之import 1 什么是模块?模块就一系统功能的集合体,在python中,一个py文件就是一个模块,比如module.py,其中模块名module2 使用模块2.1 import 导入 ...
- MYSQL中group_concat有长度限制!默认1024
在mysql中,有个函数叫"group_concat",平常使用可能发现不了问题,在处理大数据的时候,会发现内容被截取了,其实MYSQL内部对这个是有设置的,默认不设置的长度是10 ...
- 解决IE8下opacity属性失效问题
由于opacity属性存在兼容性问题,所以在IE8下,用opacity来设置元素的透明度,会失效,从而导致页面的样式问题. 在IE8及其更早的浏览器下,我们可以使用filter属性,来代替opacit ...
- mybatis的mapper接口代理使用的三个规范
1.什么是mapper代理接口方式? MyBatis之mapper代理方式.mapper代理使用的是JDK的动态代理策略 2.使用mapper代理方式有什么好处 使用这种方式可以不用写接口的实现类,免 ...
- ELK学习总结(2-6)elk的mapping
1.什么是映射 映射:创建索引的时候,预先定义字段的类型及相关属性 作用:这样会让索引建立的更加细致和完善,如:是否存储.使用何种分析器.重要级别 分类:静态映射和动态映射 2.字段类型:string ...
- Python基础数据类型之集合以及其他和深浅copy
一.基础数据类型汇总补充 list 在循环一个列表时,最好不要删除列表中的元素,这样会使索引发生改变,从而报错(可以从后向前循环删除,这样不会改变未删元素的索引). 错误示范: lis = [,,, ...
- jdk的server模式修改无效(关于client和server模式)
本机为64位操作系统,64位jdk,win10. 修改C:\Program Files\Java\jre8\lib\amd64\jvm.cfg无效. 我的文件的内容为: 原因参考如下: http:// ...
- Java线程池是如何诞生的?
时间回到2003年,那时我还是一个名不见经传的程序员,但是上级却非常看好我,他们把整个并发模块,都交给了我一个人开发. 这个星期,我必须要完成并发模块中非常重要的一个功能--线程池. 注:文末有福利 ...