【NOI2018】归程(克鲁斯卡尔重构树)

题面

洛谷

题解

我在现场竟然没有把这道傻逼题给切掉,身败名裂。

因为这题就是克鲁斯卡尔重构树的模板题啊

我就直接简单的说一下把

首先发现答案就是在只经过海拔大于\(p\)的边的情况下,所有点到\(1\)号点中最短路最小的那个点。所以预处理最短路径,构建克鲁斯卡尔重构树,直接倍增+线段树就好了。

还有一种基于离线做法的方法。

我们发现离线做法只需要按照所有询问排序,

然后利用并查集按照海拔高度从小往大合并(这个其实就是克鲁斯卡尔)

这样子就可以利用可持久并查集解决。

发现并不需要回朔时间并修改,而只需要查询历史版本的值。

因为每次增加一条边只会修改两个集合,所以可以使用一个\(vector\)存下每个点每次修改之后的父亲以及当前修改的时间,那么每次询问的时候只需要在对应的点上二分查询在目标时间的集合父亲就好了,合并使用启发式合并,保证复杂度是\(O(nlogn+Qlogn)\)。

我的代码是克鲁斯卡尔重构树

#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<cmath>
#include<algorithm>
#include<queue>
using namespace std;
#define ll long long
#define MAX 200200
#define pir pair<int,int>
#define mpi make_pair
#define fr(x) (x.first)
#define sd(x) (x.second)
#define lson (now<<1)
#define rson (now<<1|1)
inline int read()
{
int x=0;bool fl=false;char ch=getchar();
while((ch<'0'||ch>'9')&&ch!='-')ch=getchar();
if(ch=='-')fl=true,ch=getchar();
while(ch>='0'&&ch<='9')x=x*10+ch-48,ch=getchar();
return fl?-x:x;
}
struct Edge{int u,v,w,s;}E[MAX<<1];
bool operator<(Edge a,Edge b){return a.s>b.s;}
struct Line{int v,next,w,s;}e[MAX<<2];
int h[MAX<<1],cnt=1;
inline void Add(int u,int v,int w,int s){e[cnt]=(Line){v,h[u],w,s};h[u]=cnt++;}
int n,m,dis[MAX],Q,typ,S;
bool vis[MAX];
namespace SP
{
priority_queue<pir,vector<pir>,greater<pir> >Q;
void Dijkstra()
{
memset(vis,0,sizeof(vis));
while(!Q.empty())Q.pop();
Q.push(mpi(0,1));
while(!Q.empty())
{
pir u=Q.top();Q.pop();
if(vis[sd(u)])continue;vis[sd(u)]=true;
dis[sd(u)]=fr(u);
for(int i=h[sd(u)];i;i=e[i].next)
if(!vis[e[i].v])Q.push(mpi(dis[sd(u)]+e[i].w,e[i].v));
}
}
}
namespace MST
{
int f[MAX<<1],id;
int getf(int x){return x==f[x]?x:f[x]=getf(f[x]);}
void init(){for(int i=1;i<=n<<1;++i)f[i]=i;id=n;}
void Kursual()
{
init();
for(int i=1;i<=m;++i)
{
int u=getf(E[i].u),v=getf(E[i].v);
if(u==v)continue;++id;
Add(id,u,E[i].w,E[i].s);Add(id,v,E[i].w,E[i].s);
f[u]=f[v]=id;
}
}
}
int dfn[MAX<<1],low[MAX<<1],tim,ln[MAX<<1];
int p[20][MAX<<1],s[20][MAX<<1];
void dfs(int u)
{
if(u<=n)dfn[u]=++tim,ln[tim]=u;else dfn[u]=1e9;
for(int i=1;i<20;++i)p[i][u]=p[i-1][p[i-1][u]];
for(int i=1;i<20;++i)s[i][u]=min(s[i-1][u],s[i-1][p[i-1][u]]);
for(int i=h[u];i;i=e[i].next)
{
p[0][e[i].v]=u;s[0][e[i].v]=e[i].s;
dfs(e[i].v);dfn[u]=min(dfn[u],dfn[e[i].v]);
}
low[u]=tim;
}
void init(){memset(h,0,sizeof(h));cnt=1;tim=0;}
int t[MAX<<2];
void Build(int now,int l,int r)
{
if(l==r){t[now]=dis[ln[l]];return;}
int mid=(l+r)>>1;
Build(lson,l,mid);Build(rson,mid+1,r);
t[now]=min(t[lson],t[rson]);
}
int Query(int now,int l,int r,int L,int R)
{
if(L<=l&&r<=R)return t[now];
int mid=(l+r)>>1,ret=2147483647;
if(L<=mid)ret=min(ret,Query(lson,l,mid,L,R));
if(R>mid)ret=min(ret,Query(rson,mid+1,r,L,R));
return ret;
}
int Jump(int u,int r)
{
for(int i=19;~i;--i)
if(p[i][u]&&s[i][u]>r)u=p[i][u];
return u;
}
int main()
{
freopen("return.in","r",stdin);
freopen("return.out","w",stdout);
int T=read();
while(T--)
{
init();n=read();m=read();
for(int i=1;i<=m;++i)
{
int u=read(),v=read(),l=read(),s=read();
E[i]=(Edge){u,v,l,s};
Add(E[i].u,E[i].v,E[i].w,E[i].s);
Add(E[i].v,E[i].u,E[i].w,E[i].s);
}
sort(&E[1],&E[m+1]);
SP::Dijkstra();init();
memset(p,0,sizeof(p));memset(s,0,sizeof(s));
MST::Kursual();dfs(MST::id);
Build(1,1,n);
Q=read();typ=read();S=read();
int lans=0,v,p;
while(Q--)
{
v=(read()+typ*lans-1)%n+1;
p=(read()+typ*lans)%(S+1);
v=Jump(v,p);
printf("%d\n",lans=Query(1,1,n,dfn[v],low[v]));
}
}
return 0;
}

【BZOJ5415】【NOI2018】归程(克鲁斯卡尔重构树)的更多相关文章

  1. 洛谷P4768 [NOI2018]归程(克鲁斯卡尔重构树+最短路)

    传送门 前置技能,克鲁斯卡尔重构树 我们按道路的高度建一个最大生成树,然后建好克鲁斯卡尔重构树 那么我们需要知道一颗子树内到1点距离最近是多少(除此之外到子树内任何一个点都不需要代价) 可以一开始直接 ...

  2. [note]克鲁斯卡尔重构树

    克鲁斯卡尔重构树 又叫并查集重构树 大概在NOI2018之前还是黑科技 现在?烂大街了 主要是针对图上的对边有限制的一类问题 比如每次询问一个点u不能经过边权大于w的边能走到的第k大点权是多少 也就是 ...

  3. [您有新的未分配科技点][BZOJ3545&BZOJ3551]克鲁斯卡尔重构树

    这次我们来搞一个很新奇的知识点:克鲁斯卡尔重构树.它也是一种图,是克鲁斯卡尔算法求最小生成树的升级版首先看下面一个问题:BZOJ3545 Peaks. 在Bytemountains有N座山峰,每座山峰 ...

  4. 洛谷P4197 Peaks&&克鲁斯卡尔重构树学习笔记(克鲁斯卡尔重构树+主席树)

    传送门 据说离线做法是主席树上树+启发式合并(然而我并不会) 据说bzoj上有强制在线版本只能用克鲁斯卡尔重构树,那就好好讲一下好了 这里先感谢LadyLex大佬的博客->这里 克鲁斯卡尔重构树 ...

  5. 【BZOJ4242】水壶(克鲁斯卡尔重构树,BFS)

    [BZOJ4242]水壶(克鲁斯卡尔重构树,BFS) 题面 BZOJ然而是权限题. Description JOI君所居住的IOI市以一年四季都十分炎热著称. IOI市是一个被分成纵H*横W块区域的长 ...

  6. 洛谷 P1967 货车运输(克鲁斯卡尔重构树)

    题目描述 AAA国有nn n座城市,编号从 11 1到n nn,城市之间有 mmm 条双向道路.每一条道路对车辆都有重量限制,简称限重.现在有 qqq 辆货车在运输货物, 司机们想知道每辆车在不超过车 ...

  7. [模板] Kruskal算法 && 克鲁斯卡尔重构树

    克鲁斯卡尔重构树 发现没把板子放上来... 现在放一下 克鲁斯卡尔算法的正确性是利用反证法证明的. 简要地说, 就是如果不加入当前权值最小的边 \(e_1\), 那么之后加入的边和这条边会形成一个环. ...

  8. [luogu4768] [NOI2018] 归程 (Dijkstra+Kruskal重构树)

    [luogu4768] [NOI2018] 归程 (Dijkstra+Kruskal重构树) 题面 题面较长,这里就不贴了 分析 看到不能经过有积水的边,即不能经过边权小于一定值的边,我们想到了kru ...

  9. P4197 Peaks [克鲁斯卡尔重构树 + 主席树][克鲁斯卡尔重构树学习笔记]

    Problem 在\(Bytemountains\)有\(n\)座山峰,每座山峰有他的高度\(h_i\) .有些山峰之间有双向道路相连,共\(M\)条路径,每条路径有一个困难值,这个值越大表示越难走, ...

随机推荐

  1. linux安装PHP-memcache-redis扩展

    1.php memcache 扩展 http://pecl.php.net/package/memcache/3.0.8 下载文件源码 #tar zxvf memcache-3.0.8.tar#/us ...

  2. php文章tag标签的增删

    <?php session_start();   if($_POST){           $_SESSION['old']=array('one','two','three','four', ...

  3. InTelliJ 字体调整

    Java IDE 工具InTelliJ 调整字体大小 1.File -> Settings 2.左上的搜索框中输入 font. 等待自动查找结果. 3.修改size 大小

  4. HDU-2844:Coins(多重背包+二进制优化)

    链接:HDU-2844:Coins 题意:给你n个种类的钱和对应的数量,同统计一下从1到m能够凑成的钱有多少个. 题解:C[i] = 1 + 2 + 4 + ··· + 2^k + a (0 < ...

  5. OA系统与Exchange 日历打通

    目前我碰到好几个案例是希望将客户以后的OA系统与Exchange中的日历系统相结合,比如致远或者泛微的OA系统. 客户的需求如下: 1.有了OA系统 2.客户使用Outlook当邮件客户端 3.客户希 ...

  6. RNN: Feed Forward, Back Propagation Through Time and Truncated Backpropagation Through Time

    原创作品,转载请注明出处哦~ 了解RNN的前向.后向传播算法的推导原理是非常重要的,这样, 1. 才会选择正确的激活函数: 2. 才会选择合适的前向传播的timesteps数和后向传播的timeste ...

  7. Variable() placeholder() constant() 的区别

    转载来自: http://www.studyai.com/article/33e22cef42274e8a

  8. Qt类继承关系图

    分享两个资源,对于系统了解Qt框架的整体脉络很有帮助. Qt4类关系图+Qt5类关系图,PDF+JPG格式 [下载] Qt5类关系图(基于Qt5.1版),JPG格式[下载]

  9. 印度电商Snapdeal获投$1.34亿 eBay领投

    据消息人士透露,eBay领投1.337亿美元,投资印度最大在线购物网站Snapdeal,最终或有可能全权收购该网站.据悉,在此次投资中,大部分资金来自eBay. 今年1月,曾有报道称,Snapdeal ...

  10. Centos 7 zabbix 实战应用

    实际需求:公司已经有了100台服务器,现在需要使用zabbix全部监控起来. 先出个方案(规划) 常规监控:cpu,内存,磁盘,网卡 问题:怎样快速添加100台机器         方法1:使用克隆的 ...