在做Kruskal求最小生成树时,假设要通过边权\(w\)的边合并子树\(x\)和\(y\),我们新建一个方点,把两个子树接到这个方点上,并将方点的点权赋为\(w\),最终形成的二叉树就是\(Kruskal\)重构树。

这样的二叉树,是一个二叉堆,满足一个方点为根的子树内,所有方点点权都小于该点点权

统计原图上两点之间最大值的最小值时,答案就是这两点\(LCA\)的点权

\(Kruskal\)重构树很好地把一维限制放到了子树内,方便统计

对于这个题,我们以海拔为边权建一颗最大生成树,那么对应的\(Kruskal\)重构树就是一个小根堆,设一个方点\(x\)的点权为\(w\),那么以\(x\)为根的子树内,任何一对点之间经过的海拔都大于等于\(w\),于是我们只需要树上倍增找到最靠上的一个点,我们考虑该点为根的子树内的点,就取消了海拔的限制,那么只需要找\(l\)意义上最靠近1点的一个点即可。

复杂度\(O(nlogn)\)

多组数据一定要注意清空,考虑每个变量,每个数组,认真考虑!!!!!!!!!!

#include<algorithm>
#include<iostream>
#include<cstring>
#include<cstdio>
#include<queue> using namespace std; inline int rd(){
int ret=0,f=1;char c;
while(c=getchar(),!isdigit(c))f=c=='-'?-1:1;
while(isdigit(c))ret=ret*10+c-'0',c=getchar();
return ret*f;
} const int MAXN = 800005; int tot;
inline int newnode(){return ++tot;} struct Undirected_Edge{
int x,y,w;
bool operator<(const Undirected_Edge &rhs)const {
return w>rhs.w;
}
}ue[MAXN];
struct Edge{
int ecnt,head[MAXN],nxt[MAXN],to[MAXN],h[MAXN],l[MAXN];
Edge(){ecnt=1;}
void clear(){
ecnt=1;
memset(head,0,sizeof(head));
}
void add(int x,int y,int _l=0,int _h=0){
nxt[++ecnt] = head[x];
to[ecnt] = y;
h[ecnt] = _h;
l[ecnt] = _l;
head[x] = ecnt;
}
}e,t; int n,m; struct Uno{
int fa[MAXN];
void init(int x){for(int i=1;i<=x;i++)fa[i]=i;}
int fnd(int x){return x==fa[x]?x:fa[x]=fnd(fa[x]);}
void cat(int x,int y){x=fnd(x);y=fnd(y);if(x==y)return;fa[x]=y;}
}U;
struct Node{
int id,v;
Node(int _id=0,int _v=0){id=_id;v=_v;}
bool operator <(const Node &rhs)const{return v>rhs.v;}
}top;
priority_queue<Node> Q;
int vis[MAXN],dis[MAXN];
void dij(){
Q.push(Node(1,0));dis[1]=0;
while(!Q.empty()){
top=Q.top();Q.pop();
int mnid=top.id,mn=top.v;
if(vis[mnid]||mn!=dis[mnid])continue;
vis[mnid]=1;
for(int i=e.head[mnid];i;i=e.nxt[i]){
int v=e.to[i];
if(dis[v]<=mn+e.l[i])continue;
dis[v]=mn+e.l[i];
Q.push(Node(v,dis[v]));
}
}
} int low[MAXN],val[MAXN],f[MAXN][32];
void init(){
U.init(n<<1);tot=n;
memset(dis,0x3f,sizeof(dis));
memset(vis,0,sizeof(vis));
memset(low,0x7f,sizeof(low));
memset(val,0,sizeof(val));
memset(f,0,sizeof(f));
e.clear();t.clear();
} void dfs(int x,int pre){
f[x][0]=pre;low[x]=dis[x];
for(int i=t.head[x];i;i=t.nxt[i]){
int v=t.to[i];
if(v==pre)continue;
dfs(v,x);low[x]=min(low[x],low[v]);
}
} int q,k,s;
int lastans; void solve(){
lastans=0;//fff
int x,y,u,v,l,h,cur;
for(int i=1;i<=m;i++){
x=rd();y=rd();l=rd();h=rd();
e.add(x,y,l,h);e.add(y,x,l,h);
ue[i].x = x;ue[i].y = y;ue[i].w = h;
}
dij();
sort(ue+1,ue+1+m);
int Edge_cnt=0;
for(int i=1;i<=m;i++){
x=ue[i].x,y=ue[i].y,h=ue[i].w;
u=U.fnd(x);v=U.fnd(y);
if(u==v)continue;
cur=newnode();Edge_cnt++;
U.cat(u,cur);U.cat(v,cur);
t.add(u,cur);t.add(cur,u);
t.add(v,cur);t.add(cur,v);
val[cur]=h;
if(Edge_cnt==n-1)break;
}
int rt=U.fnd(1);
dfs(rt,rt);
for(int j=1;(1<<j)<=tot;j++){
for(int i=1;i<=tot;i++){
f[i][j]=f[f[i][j-1]][j-1];
}
}
q=rd();k=rd();s=rd();
for(int i=1;i<=q;i++){
x=rd();y=rd();
u=(x+k*lastans-1)%n+1;
v=(y+k*lastans)%(s+1);
for(int i=30;i>=0;i--){
if(val[f[u][i]]<=v)continue;
u=f[u][i];
}
printf("%d\n",low[u]);
lastans=low[u];
}
} int T; signed main(){
T=rd();
while(T--){
n=rd();m=rd();
init();
solve();
}
return 0;
}

[BZOJ] 5415: [Noi2018]归程的更多相关文章

  1. BZOJ 5415: [Noi2018]归程(kruskal重构树)

    解题思路 \(NOI2018\)的\(Day1\) \(T1\),当时打网络赛的时候不会做.学了一下\(kruskal\)重构树后发现问题迎刃而解了.根据\(kruskal\)的性质,如果要找从\(u ...

  2. 【刷题】BZOJ 5415 [Noi2018]归程

    www.lydsy.com/JudgeOnline/upload/noi2018day1.pdf Solution 考试的时候打的可持久化并查集,没调出来QAQ 后面知道了kruskal重构树这个东西 ...

  3. [LOJ 2718][UOJ 393][BZOJ 5415][NOI 2018]归程

    [LOJ 2718][UOJ 393][BZOJ 5415][NOI 2018]归程 题意 给定一张无向图, 每条边有一个距离和一个高度. 再给定 \(q\) 组可能在线的询问, 每组询问给定一个点 ...

  4. [NOI2018]归程 kruskal重构树

    [NOI2018]归程 LG传送门 kruskal重构树模板题. 另一篇文章里有关于kruskal重构树更详细的介绍和更板子的题目. 题意懒得说了,这题的关键在于快速找出从查询的点出发能到达的点(即经 ...

  5. [洛谷P4768] [NOI2018]归程 (kruskal重构树模板讲解)

    洛谷题目链接:[NOI2018]归程 因为题面复制过来有点炸格式,所以要看题目就点一下链接吧\(qwq\) 题意: 在一张无向图上,每一条边都有一个长度和海拔高度,小\(Y\)的家在\(1\)节点,并 ...

  6. NOI2018 D1T1 [NOI2018]归程 解题报告

    P4768 [NOI2018]归程 题目描述 本题的故事发生在魔力之都,在这里我们将为你介绍一些必要的设定. 魔力之都可以抽象成一个 \(n\) 个节点.\(m\) 条边的无向连通图(节点的编号从 \ ...

  7. BZOJ_5415_[Noi2018]归程_kruscal重构树+倍增+最短路

    BZOJ_5415_[Noi2018]归程_kruscal重构树+倍增 Description www.lydsy.com/JudgeOnline/upload/noi2018day1.pdf 好久不 ...

  8. 题解 NOI2018 归程

    题解 NOI2018 归程 题意 本题的故事发生在魔力之都,在这里我们将为你介绍一些必要的设定. 魔力之都可以抽象成一个 n 个节点.m 条边的无向连通图(节点的编号从 1 至 n).我们依次用 l, ...

  9. [NOI2018]归程(kruscal重构树)

    [NOI2018]归程 题面太长辣,戳这里 模拟赛上写了一个spfa (关于spfa,它已经死了),然后一个st表水完暴力跑路.考后说是Kruscal重构树或者可持久化并查集???这都是些什么东西.不 ...

随机推荐

  1. 关于HTML5用SVG画图

    SVG在HTML5中的应用 SVG(Scalable Vector Graphics)是用来绘制矢量图的HTML5标签.只要定义好XML属性就能够获得与其一致的图像元素. 使用SVG之前先将标签加入到 ...

  2. PAT甲级——1106 Lowest Price in Supply Chain(BFS)

    本文同步发布在CSDN:https://blog.csdn.net/weixin_44385565/article/details/90444872 1106 Lowest Price in Supp ...

  3. Codeforces 1105D(双层广搜)

    要点 题意:可以拐弯,即哈密顿距离 注意不可以直接一个一个搜,这过程中会把下一轮的标记上,导致同一轮的其它点没能正常完成应有的搜索 因此采用双层广搜,把同一轮先都出队列再的一起搜 #include & ...

  4. linux yum 安装

    ################## http://rpm.pbone.net/ 下载下来的包放到本地yum源中,然后在这个目录下面重新生成依赖关系就可以使用yum包来完成安装了 tt 1. 生成依赖 ...

  5. B -- RE:从零开始的异世界生活 线段树

    http://www.ifrog.cc/acm/problem/1117?contest=1016&no=1 其实我是第一次这样用线段树. 首先把所有出现过的数字全部离散化.那么数字就是从[1 ...

  6. rabbitmq 不发送ack消息如何处理:rabbitmq可靠发送的自动重试机制

    转载地址:http://www.jianshu.com/p/6579e48d18ae http://www.jianshu.com/p/4112d78a8753 接这篇 在上文中,主要实现了可靠模式的 ...

  7. Eclipse 在桌面创建快捷方式打开工作区

    创建eclipse快捷的方式,并Copy到桌面. 打开快捷方式的属性窗口,在[目标]栏,增加参数 –data 你的工作区目录

  8. Lodop套打

    记录一下Lodop套打模板 实现打印功能需电脑已经连接打印机(打印什么类型的东西就连接相应的打印机 (普通大打印机 打印标签 打印发票各种打印机))和已经安装好lodop控件 控件可到官网进行下载 h ...

  9. AJPFX关于Java中的集合

    ava API中所用的集合类,都是实现了Collection接口,他的一个类继承结构如下: Collection<--List<--Vector Collection<--List& ...

  10. str中的join方法,fromkeys(),set集合,深浅拷贝(重点)

    一丶对之前的知识点进行补充 1.str中的join方法.把列表转换成字符串 # 将列表转换成字符串,每个元素之间用_拼接 s = "_".join(["天",& ...