博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
NOI2002银河英雄传说——带权并查集
阅读量:6274 次
发布时间:2019-06-22

本文共 1274 字,大约阅读时间需要 4 分钟。

题目:

关键点在于存下每个点的位置。

自己糊涂的地方:位置是相对于谁的位置?

  因为每次给一个原来是fa的点赋位置时,赋的位置与此点的新fa有关,故很容易想到在 find() fa 时顺便更新 位置 们。

  所以每次存的位置应该是新fa的集的大小——不能加上新fa自己的位置,那样就混乱了。

  要点在于若想在 find() 更新 fa 时同步更新 pos,则存的位置必须是 相对于自己当前 fa 的位置!

自己的难点:怎样做到递归同步更新pos?

  递归的要点是先运行一遍更深层的,则更深层的地方就被赋好了值,就可以更新当前层了。

  所以不能按平常写法:return fa[a]=find(fa[a]);而要在 find() 和 return 中间插一个更新pos的语句!

#include
#include
using namespace std;int t,a,b,fa[30005],pos[30005],siz[30005];//pos是相对自己的fa的位置(无自己,有fa),char c; //只有这样才能更新fa时同步更新pos int find(int a){ if(fa[a]==a)return a; int k=find(fa[a]); pos[a]+=pos[fa[a]]; return fa[a]=k;}int main(){ for(int i=1;i<=30000;i++)fa[i]=i,siz[i]=1; scanf("%d",&t); while(t--) { scanf(" %c%d%d",&c,&a,&b); if(c=='M') { int u=find(a); int v=find(b); fa[u]=v; pos[u]=siz[v]; siz[v]+=siz[u]; } if(c=='C') { int u=find(a); int v=find(b); if(u!=v)printf("-1\n"); else { int k=pos[a]-pos[b]; if(k<0)k=-k; printf("%d\n",k-1); } } } return 0;}

 

转载于:https://www.cnblogs.com/Narh/p/8299367.html

你可能感兴趣的文章
大佬是怎么思考设计MySQL优化方案的?
查看>>
<三体> 给岁月以文明, 给时光以生命
查看>>
Android开发 - 掌握ConstraintLayout(九)分组(Group)
查看>>
springboot+logback日志异步数据库
查看>>
Typescript教程之函数
查看>>
Android 高效安全加载图片
查看>>
vue中数组变动不被监测问题
查看>>
3.31
查看>>
类对象定义 二
查看>>
收费视频网站Netflix:用户到底想要“点”什么?
查看>>
MacOS High Sierra 12 13系统转dmg格式
查看>>
关于再次查看已做的多选题状态逻辑问题
查看>>
动态下拉菜单,非hover
查看>>
政府安全资讯精选 2017年第十六期 工信部发布关于规范互联网信息服务使用域名的通知;俄罗斯拟建立备用DNS;Google打击安卓应用在未经同意情况下收集个人信...
查看>>
简单易懂的谈谈 javascript 中的继承
查看>>
多线程基础知识
查看>>
iOS汇编基础(四)指针和macho文件
查看>>
Laravel 技巧锦集
查看>>
Android 使用 ViewPager+RecyclerView+SmartRefreshLayout 实现顶部图片下拉视差效果
查看>>
Flutter之基础Widget
查看>>