NOI 2018 归程 (Kruskal重构树)
题目大意:太长了,略
Kruskal重构树,很神奇的一个算法吧
如果两个并查集被某种条件合并,那么这个条件作为一个新的节点连接两个并查集
那么在接下来的提问中,如果某个点合法,它的所有子节点也都合法,即子节点的限制少于父节点
#include <cstdio>
#include <algorithm>
#include <cstring>
#include <queue>
#define inf 0x3f3f3f3f
#define ll long long
#define il inline
#define N 400100
#define M 800100
using namespace std;
//re
int T,cte,ctb,n,m,tot;
int head[N],hbt[M],fa[M],ff[M][],dis[N],use[N],mi[M],hei[M];
struct EDGE{int to,nxt,val;}edge[M];
struct Krs{int x,y,alt;}krs[N];
struct BT{int to,nxt;}bt[M*];
struct node{int id,dis;};
int cmpk(Krs s1,Krs s2){return s1.alt>s2.alt;}
il node ins(int x1,int x2){node kk;kk.id=x1,kk.dis=x2;return kk;}
bool operator<(const node &s1,const node &s2){return s1.dis>s2.dis;}
int find_fa(int x){
int fx=fa[x],pre;while(fx!=fa[fx])fx=fa[fx];
while(fa[x]!=fx){pre=fa[x],fa[x]=fx,x=pre;}
return fx;
}
int gc(){
int rett=,fh=;char p=getchar();
while(p<''||p>'') {if(fh=='-')fh=-;p=getchar();}
while(p>=''&&p<='') {rett=(rett<<)+(rett<<)+p-'';p=getchar();}
return rett*fh;
}
void clr()
{
cte=ctb=tot=;
memset(fa,,sizeof(fa));memset(ff,,sizeof(ff));
memset(krs,,sizeof(krs));memset(bt,,sizeof(bt));
memset(mi,0x3f,sizeof(mi));memset(edge,,sizeof(edge));
memset(head,-,sizeof(head));memset(hbt,-,sizeof(hbt));
}
void abt(int u,int v){
ctb++;bt[ctb].to=v;
bt[ctb].nxt=hbt[u],hbt[u]=ctb;
}
void ae(int u,int v,int w){
cte++;edge[cte].to=v,edge[cte].val=w;
edge[cte].nxt=head[u],head[u]=cte;
}
void dfs_bt(int x)
{
mi[x]=dis[x];
for(int j=hbt[x];j!=-;j=bt[j].nxt){
int v=bt[j].to;
if(v==ff[x][]) continue;
ff[v][]=v,ff[v][]=x;
dfs_bt(v);
mi[x]=min(mi[x],min(mi[v],dis[v]));
}
}
void get_multip(){
for(int j=;j<=;j++)
for(int i=;i<=tot;i++)
ff[i][j] = ff[ ff[i][j-] ][j-];
}
int multi(int x,int p){
for(int j=;j>=;j--){
if(hei[ff[x][j]]>p) x=ff[x][j];
}return x;
}
void dijkstra()
{
priority_queue<node>que;
memset(dis,0x3f,sizeof(dis));
memset(use,,sizeof(use));
dis[]=,que.push(ins(,));
while(!que.empty()){
node ss=que.top();que.pop();
if(use[ss.id]) continue;
use[ss.id]=;int x=ss.id;
for(int j=head[x];j!=-;j=edge[j].nxt){
int v=edge[j].to;
if(dis[v]>dis[x]+edge[j].val){
dis[v]=dis[x]+edge[j].val;
if(!use[v]) que.push(ins(v,dis[v]));
}
}
}
}
void Kruskal()
{
int fx,fy,sum=;tot=n;
for(int i=;i<=*n;i++) fa[i]=i;
sort(krs+,krs+m+,cmpk);
for(int i=;i<=m;i++){
fx=find_fa(krs[i].x),fy=find_fa(krs[i].y);
if(fx==fy) continue;
abt(++tot,fx),abt(tot,fy);
hei[tot]=krs[i].alt,sum++;
fa[fx]=tot,fa[fy]=tot;
if(sum==n-) break;
}hei[]=-;
dfs_bt(tot);
get_multip();
}
int solve(int x,int p)
{
int fx=multi(x,p);
return mi[fx];
} int main()
{
//freopen("data.in","r",stdin);
scanf("%d",&T);
while(T--)
{ n=gc(),m=gc();clr();
int x,y,w,z,lst=;
for(int i=;i<=m;i++)
{
x=gc(),y=gc(),w=gc(),z=gc();
ae(x,y,w),ae(y,x,w);
krs[i].x=x,krs[i].y=y,krs[i].alt=z;
}
dijkstra();
Kruskal();
int q,k,s;
q=gc(),k=gc(),s=gc();
for(int i=;i<=q;i++)
{
x=gc(),w=gc();
x=(x+k*lst-)%n+;
w=(w+k*lst)%(s+);
lst=solve(x,w);
printf("%d\n",lst);
} }
return ;
}
NOI 2018 归程 (Kruskal重构树)的更多相关文章
- NOI Day1T1归程(Kruskal重构树+Dijkstra)
NOI Day1T1归程(Kruskal重构树+Dijkstra) 题目 洛谷题目传送门 题解 其实我不想写......,所以...... 挖个坑......我以后一定会补的 luogu的题解讲的还是 ...
- BZOJ5415[Noi2018]归程——kruskal重构树+倍增+堆优化dijkstra
题目描述 本题的故事发生在魔力之都,在这里我们将为你介绍一些必要的设定. 魔力之都可以抽象成一个 n 个节点.m 条边的无向连通图(节点的编号从 1 至 n).我们依次用 l,a 描述一条边的长度.海 ...
- [NOI2018]归程 kruskal重构树
[NOI2018]归程 LG传送门 kruskal重构树模板题. 另一篇文章里有关于kruskal重构树更详细的介绍和更板子的题目. 题意懒得说了,这题的关键在于快速找出从查询的点出发能到达的点(即经 ...
- [洛谷P4768] [NOI2018]归程 (kruskal重构树模板讲解)
洛谷题目链接:[NOI2018]归程 因为题面复制过来有点炸格式,所以要看题目就点一下链接吧\(qwq\) 题意: 在一张无向图上,每一条边都有一个长度和海拔高度,小\(Y\)的家在\(1\)节点,并 ...
- 洛谷P4768 [NOI2018]归程(Kruskal重构树)
题意 直接看题目吧,不好描述 Sol 考虑暴力做法 首先预处理出从$1$到每个节点的最短路, 对于每次询问,暴力的从这个点BFS,从能走到的点里面取$min$ 考虑如何优化,这里要用到Kruskal重 ...
- LOJ.2718.[NOI2018]归程(Kruskal重构树 倍增)
LOJ2718 BZOJ5415 洛谷P4768 Rank3+Rank1无压力 BZOJ最初还不是一道权限题... Update 2019.1.5 UOJ上被hack了....好像是纯一条链的数据过不 ...
- #2718. 「NOI2018」归程 kruskal重构树
链接 https://loj.ac/problem/2718 思路 我们希望x所在的连通块尽量的大,而且尽量走高处 离线的话可以询问排序,kruskal过程中更新答案 在线就要用kruskal重构树 ...
- loj2718 「NOI2018」归程[Kruskal重构树+最短路]
关于Kruskal重构树可以翻阅本人的最小生成树笔记. 这题明显裸的Kruskal重构树. 然后这题限制$\le p$的边不能走,实际上就是要满足走最小边权最大的瓶颈路,于是跑最大生成树,构建Krus ...
- BZOJ 5415: [Noi2018]归程(kruskal重构树)
解题思路 \(NOI2018\)的\(Day1\) \(T1\),当时打网络赛的时候不会做.学了一下\(kruskal\)重构树后发现问题迎刃而解了.根据\(kruskal\)的性质,如果要找从\(u ...
随机推荐
- BZOJ 1901 Dynamic Rankings (整体二分+树状数组)
题目大意:略 洛谷传送门 这道题在洛谷上数据比较强 貌似这个题比较常见的写法是树状数组套主席树,动态修改 我写的是整体二分 一开始的序列全都视为插入 对于修改操作,把它拆分成插入和删除两个操作 像$C ...
- [USACO4.2] 草地排水 Drainage Ditches (最大流)
题目背景 在农夫约翰的农场上,每逢下雨,贝茜最喜欢的三叶草地就积聚了一潭水.这意味着草地被水淹没了,并且小草要继续生长还要花相当长一段时间.因此,农夫约翰修建了一套排水系统来使贝茜的草地免除被大水淹没 ...
- [poj 3666] Making the Grade (离散化 线性dp)
今天的第一题(/ω\)! Description A straight dirt road connects two fields on FJ's farm, but it changes eleva ...
- JavaScript CSS 实现简单的 TAB 标签切换
使用CSS隐藏所有tab页,然后使用JavaScript给选中的元素对应ID的tab页设置class="active"类来显示该元素,以此实现tab切换. 如鼠标放置到shwww时 ...
- 小学生都能学会的python(一)2018.9.3
一,小学生第一天 1,认识和了解python python的创始⼈为吉多·范罗苏姆(Guido van Rossum). python是一门解释性语言 弱类型语言 优点:(1).Python的定位是 ...
- Hibernate的事务机制
http://www.cnblogs.com/jerryxing/archive/2012/04/24/2468999.html http://www.cnblogs.com/xiaofeilee/p ...
- centos 如何执行.bin 文件??
默认下载,或者上传到 linux 上的 .bin 文件都是白色的,不能被执行. 想要用 ./ 命令去执行,需要先给这个 bin 文件赋权限 chmod u+x 文件名(全名称) 执行完成这个命令,可以 ...
- windows FTP自动下载脚本
新建ftp.dat文件,内容如下 open xxx.xxx.xxx.xxxusernamepasswordcd lcd x:\filesbinaryhashmget *.*bye 新建run.b ...
- SVN中各种符号箭头含义
黄色感叹号(有冲突): -- 这是有冲突了,冲突就是说你对某个文件进行了修改,别人也对这个文件进行了修改,别人抢在你提交之前先提交了,这时你再提交就会被提示发生冲突,而不允许 你提交,防止你的提交覆盖 ...
- CF409C Magnum Opus
CF409C Magnum Opus 题意翻译 题目背景 愚人节题目,题面似乎是一位名叫Nicolas Flamel的炼金术士用拉丁文写的某种物质的配方,结合谷歌尝试翻译了一下: 吾友: 哲人石所言不 ...