[BZOJ3694]最短路(并查集)
2016-04-27 15:56
411 查看
题目描述
传送门题解
首先建出最短路径树,考虑对于每一条非树边(u,v),令t=lca(u,v),则u,v及以上的点、t以下的点都会由这条非树边多出一条路径。设disi为1~i的最短路径,那么对于每个u,v及以上的点、t以下的点,新增的这条路径的长度为disu+disv+len(u,v)-disx,又因为对于每一个x disx是恒定的,那么我们的目标就是求出能用来更新x的最小的非树边。链剖的思路很显然,不过并查集也可以做。那么令每一条非树边的权值为disu+disv+len(u,v)-disx,排好序后一旦有一个点被更新了,就不会被其他的值更新,那么就可以记录一下每一个已经被更新过了的链顶,用并查集维护。
代码
#include<algorithm> #include<iostream> #include<cstring> #include<cstdio> using namespace std; const int max_n=4e3+5; const int max_m=1e5+5; const int max_e=max_m*2; int n,m,a,b,l,t,cnt; int tot,point[max_n],next[max_e],v[max_e],c[max_e]; int h[max_n],father[max_n],dis[max_n],f[max_n],belong[max_n]; struct hp{int x,y,l,len;}edge[max_m]; inline void add(int x,int y,int z){++tot;next[tot]=point[x];point[x]=tot;v[tot]=y;c[tot]=z;} inline void dfs(int x,int fa,int dep){ h[x]=dep; father[x]=fa; for (int i=point[x];i;i=next[i]) if (v[i]!=fa){ dis[v[i]]=dis[x]+c[i]; dfs(v[i],x,dep+1); } } inline int cmp(hp a,hp b){return a.len<b.len;} inline int find(int x){if (x==f[x]) return x;else return f[x]=find(f[x]);} int main(){ scanf("%d%d",&n,&m); for (int i=1;i<=m;++i){ scanf("%d%d%d%d",&a,&b,&l,&t); if (t==1) add(a,b,l),add(b,a,l); else edge[++cnt].x=a,edge[cnt].y=b,edge[cnt].l=l; } dfs(1,0,1); for (int i=1;i<=cnt;++i) edge[i].len=edge[i].l+dis[edge[i].x]+dis[edge[i].y]; sort(edge+1,edge+cnt+1,cmp); for (int i=1;i<=n;++i) f[i]=i; for (int i=1;i<=cnt;++i){ int u=edge[i].x,t=edge[i].y,f1=find(u),f2=find(t),lastu=0,lastt=0; while (f1!=f2){ if (h[f1]<h[f2]) swap(u,t),swap(f1,f2),swap(lastu,lastt); if (!belong[u]){ belong[u]=i; if (lastu) f[lastu]=u; } else if (lastu) f[lastu]=f1; lastu=f1;u=father[lastu];f1=find(u); } } if (belong[2]) printf("%d",edge[belong[2]].len-dis[2]); else printf("-1"); for (int i=3;i<=n;++i) if (belong[i]) printf(" %d",edge[belong[i]].len-dis[i]); else printf(" -1"); }
总结
“最短路”一定是树,不可能形成环。(想一想为什么?)相关文章推荐
- Django error信息邮件通知功能配置部署
- 已知圆心,半径,角度,求圆上的点坐标
- 判断js对象是否拥有某一个属性的js代码
- spark-submit使用及说明
- UIImage和UIColor的转换
- getting-started-with-mqtt
- Nginx 之三:nginx服务器模块、web请求处理机制及事件驱动模型、进程功能和进程间通信
- jwplayer 网页在线播放插件
- 类属性和栈模拟代码
- bkdr hash
- IE11 F12工具报错
- git 高级使用
- Android手势密码实现方案
- 早期练习正则与字符串的一些记录
- 会话管理之session技术
- IMP
- 虚拟机上安装ArchLinux笔记
- jquery选择器案例
- 记录一次Android交叉编译ffmpeg排查错误
- java反射中getDeclaredField和getField的区别