思路:

Kruskal求最大生成树+倍增LCA

// by SiriusRen
#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
#define N 105000
int n,m,tot=0,xx,yy,zz,ans;
int first[N],v[N*10],next[N*10],w[N*10],f[N],dep[N],fa[N][20],minn[N][20];
int find(int x){return x==f[x]?x:f[x]=find(f[x]);}
struct EDGE{int from,to,weight;}Edge[50500];
void add(int x,int y,int z){
w[tot]=z,v[tot]=y;
next[tot]=first[x];
first[x]=tot++;
}
bool cmp(EDGE x,EDGE y){return x.weight>y.weight;}
void dfs(int x){
for(int j=1;j<=18;j++){
fa[x][j]=fa[fa[x][j-1]][j-1];
minn[x][j]=min(minn[x][j-1],minn[fa[x][j-1]][j-1]);
}
for(int i=first[x];~i;i=next[i])
if(dep[v[i]]==-1){
dep[v[i]]=dep[x]+1;
fa[v[i]][0]=x;minn[v[i]][0]=w[i];
dfs(v[i]);
}
}
int lca(int x,int y){
int ans=0x3fffffff;
if(dep[x]<dep[y])swap(x,y);
for(int i=18;i>=0;i--)if(dep[x]>=dep[y]+(1<<i))ans=min(ans,minn[x][i]),x=fa[x][i];
if(x==y)return ans;
for(int i=18;i>=0;i--)
if(fa[x][i]!=fa[y][i]){
ans=min(ans,min(minn[x][i],minn[y][i]));
x=fa[x][i];y=fa[y][i];
}
return min(ans,min(minn[x][0],minn[y][0]));
}
int main(){
scanf("%d%d",&n,&m);
for(int i=1;i<=n;i++)f[i]=i;
memset(dep,-1,sizeof(dep));
memset(minn,0x3f,sizeof(minn));
memset(first,-1,sizeof(first));
for(int i=1;i<=m;i++){
scanf("%d%d%d",&xx,&yy,&zz);
Edge[i].from=xx;Edge[i].to=yy;Edge[i].weight=zz;
}
sort(Edge+1,Edge+1+m,cmp);
for(int i=1;i<=m;i++)
if(find(Edge[i].from)!=find(Edge[i].to)){
f[find(Edge[i].from)]=find(Edge[i].to);
add(Edge[i].from,Edge[i].to,Edge[i].weight);
add(Edge[i].to,Edge[i].from,Edge[i].weight);
}
dep[find(1)]=0;dfs(find(1));
scanf("%d",&m);
while(m--){
scanf("%d%d",&xx,&yy);
if(~dep[xx]&&~dep[yy])printf("%d\n",lca(xx,yy));
else puts("-1");
}
}

队长讲了还有一中很奇怪的方法可以乱搞。

就是:Bling 并查集!

我们可以想到Kruskal进行的过程中是把两个连通块连起来,中间连的边一定比连通块里面的边要小。

那么我们可以考虑按秩合并。。可以证明这样树的高度是log的。

然后直接暴力求LCA即可

网上是这么说的:

启发式并查集,就是维护每个集合的深度,在合并两个集合的时候把小的那个集合挂在大集合下。

在此题中呢,求最大生成树的同时,不把新加入的一条边作为计算答案的树,而是把两个集合的祖先加入树中,边权就是原来边的两个边权。看到这,不禁产生了疑问,树的边权和形态与求出的最大生成树都不一样,为啥能做???其实没有关系,因为新加入的边不影响

原来集合中两点的答案,合并的两个集合中的点合并后肯定要经过原来这条边,那我把祖先接起来用原来边的边权也是一样的。

但是这么做,由于使用了启发式合并,那么最后新的树高度可以证明不会超过logn(其实我也不会证大笑),那么我们不用倍增处理这棵树,直接暴力求lca即可,不仅代码短,而且常数小!!!

// by SiriusRen
#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
#define N 105000
int n,m,tot=0,xx,yy,zz;
int first[N],v[N*10],next[N*10],w[N*10],f[N],dep[N],fa[N],size[N],minn[N];
int find(int x){return x==f[x]?x:f[x]=find(f[x]);}
struct EDGE{int from,to,weight;}Edge[50500];
void add(int x,int y,int z){w[tot]=z,v[tot]=y;next[tot]=first[x];first[x]=tot++;}
bool cmp(EDGE x,EDGE y){return x.weight>y.weight;}
void dfs(int x){
for(int i=first[x];~i;i=next[i])
if(dep[v[i]]==-1){
dep[v[i]]=dep[x]+1;
fa[v[i]]=x;minn[v[i]]=w[i];
dfs(v[i]);
}
}
void lca(int x,int y){
int ans=0x3fffffff;
if(dep[x]>dep[y])swap(x,y);
while(dep[x]!=dep[y])ans=min(minn[y],ans),y=fa[y];
while(x!=y){
ans=min(ans,min(minn[x],minn[y]));
x=fa[x];y=fa[y];
}
printf("%d\n",ans);
return;
}
int main(){
scanf("%d%d",&n,&m);
for(int i=1;i<=n;i++)size[i]=1;
for(int i=1;i<=n;i++)f[i]=i;
memset(dep,-1,sizeof(dep));
memset(first,-1,sizeof(first));
memset(minn,0x3f,sizeof(minn));
for(int i=1;i<=m;i++){
scanf("%d%d%d",&xx,&yy,&zz);
Edge[i].from=xx;Edge[i].to=yy;Edge[i].weight=zz;
}
sort(Edge+1,Edge+1+m,cmp);
for(int i=1;i<=m;i++){
int fx=find(Edge[i].from),fy=find(Edge[i].to);
if(fx!=fy){
if(size[fx]>size[fy])swap(fx,fy);
f[fx]=fy;size[fy]+=fx;
add(fx,fy,Edge[i].weight);add(fy,fx,Edge[i].weight);
}
}
dep[find(1)]=0;dfs(find(1));
scanf("%d",&m);
while(m--){
scanf("%d%d",&xx,&yy);
if(~dep[xx]&&~dep[yy])lca(xx,yy);
else puts("-1");
}
}

NOIP2013 D1T3 货车运输 倍增LCA OR 并查集按秩合并的更多相关文章

  1. 【bzoj4668】冷战 并查集按秩合并+朴素LCA

    题目描述 1946 年 3 月 5 日,英国前首相温斯顿·丘吉尔在美国富尔顿发表“铁幕演说”,正式拉开了冷战序幕. 美国和苏联同为世界上的“超级大国”,为了争夺世界霸权,两国及其盟国展开了数十年的斗争 ...

  2. 洛谷P3379lca,HDU2586,洛谷P1967货车运输,倍增lca,树上倍增

    倍增lca板子洛谷P3379 #include<cstdio> struct E { int to,next; }e[]; ],anc[][],log2n,deep[],n,m,s,ne; ...

  3. NOIP2013 D1T3 货车运输

    [NOIP2013T3]货车运输 背景 noip2013day1 描述 A 国有 n 座城市,编号从 1 到 n,城市之间有 m 条双向道路.每一条道路对车辆都有重 量限制,简称限重.现在有 q 辆货 ...

  4. 洛谷P1967货车运输——倍增LCA

    题目:https://www.luogu.org/problemnew/show/P1967 就是倍增LCA的裸题,注意一些细节即可. 代码如下: #include<iostream> # ...

  5. NOIP2013 D1T3 货车运输 zz耻辱记

    目录 先来证明下lemma: 图上2点间最小边权最大的路径一定在MST上 感性理解下: 每次kruskal algo都连接最大的不成环边 此时有2个未联通的联通块被连起来. 那么考虑u, v两点的联通 ...

  6. xsy 2018 【NOIP2013】货车运输

    [NOIP2013]货车运输 Description A 国有n座城市,编号从1到n,城市之间有m条双向道路.每一条道路对车辆都有重量限制,简称限重.现在有q辆货车在运输货物,司机们想知道每辆车在不超 ...

  7. Luogu 2245 星际导航(最小生成树,最近公共祖先LCA,并查集)

    Luogu 2245 星际导航(最小生成树,最近公共祖先LCA,并查集) Description sideman做好了回到Gliese 星球的硬件准备,但是sideman的导航系统还没有完全设计好.为 ...

  8. LCA tarjan+并查集POJ1470

    LCA tarjan+并查集POJ1470 https://www.cnblogs.com/JVxie/p/4854719.html 不错的一篇博客啊,让我觉得LCA这么高大上的算法不是很难啊,嘻嘻嘻 ...

  9. NOIP2013 货车运输 倍增

    问题描述 A 国有 n 座城市,编号从 1 到 n,城市之间有 m 条双向道路.每一条道路对车辆都有重量限制,简称限重.现在有 q 辆货车在运输货物,司机们想知道每辆车在不超过车辆限重的情况下,最多能 ...

随机推荐

  1. JSP_内置对象_response

    response对象: response对象包含了相应客户请求的有关信息,但在JSP中很少直接用到它,它是HttpServletResponse类的实例.response对象具有页面作用域,即访问一个 ...

  2. webstorm for mac

    mac上使用webstrom,破解的方法 参见博客:Webstorm 破解2016.1 for mac 上面的说明有点问题,博主的写的有点问题.应该是1.先打开到注册页面:2.再关闭webstrom; ...

  3. Django 模型层(标签、过滤器、模板的继承与导入)

    过滤器/自定义过滤器 模板语法中的过滤器类似于python中的内置方法,在我们把数据从后端通过rander传入到前端html文件中之后,在前端我们可以通过模板语法,对传入的数据再进行以通骚操作. 首先 ...

  4. eas之dep的前置脚本和后置脚本

    dep的前置脚本和后置脚本,什么时候写,是这样解释的:    前置脚本是在方法前执行,后置脚本是在方法后执行    1.比如保存扩展,如果你要在保存前校验某个字段的值,你要在前置脚本中写,如果要保存后 ...

  5. 【剑指Offer】49、把字符串转换成整数

      题目描述:   将一个字符串转换成一个整数(实现Integer.valueOf(string)的功能,但是string不符合数字要求时返回0),要求不能使用字符串转换整数的库函数. 数值为0或者字 ...

  6. 八进制、十进制、操作符(day04)

    把二进制表示的数字从右向左每三个数位分成 一组,每组用一个0到7之间的数字替换. 这个替换结果叫做数字的八进制表示方式 (八进制) 可以直接在程序里用八进制方式表示数字, 这种数字必须以0做开头 可以 ...

  7. Java设计模式之 — 单例(Singleton)

    转载请注明出处:http://blog.csdn.net/guolin_blog/article/details/8860649 写软件的时候经常需要用到打印日志功能,可以帮助你调试和定位问题,项目上 ...

  8. python 图片滑动窗口

    METHOD #1: No smooth, just scaling. def pyramid(image, scale=1.5, minSize=(30, 30)): # yield the ori ...

  9. Request中通过文件流获取文件

    第一次写博客,希望能帮到以后接触到这里的同学,废话不多说,面对疾风吧. /** * 获取文件相信信息 * @param request HttpServletRequest实例 * @param im ...

  10. Solr Update插件自定义Update Chain按条件更新索引

    背景:基于call客,来电和跟进记录等多个数据来源的用户文档,需要在更新是判断首来源的时间. 如对电话号码11xxxx来说,来电时间是今天,call客时间是昨天,而call客数据又可能因为网络原因晚上 ...