NOIP2013 货车运输(最大生成树,倍增)

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

首先肯定是跑一个最大生成数辣~跑完以后倍增lca即可。注意kruskal的写法,并查集一定要写对!

#include <cstdio>
#include <algorithm>
using namespace std; const int maxn=1e4+5, maxm=5e4+5, INF=1e9;
struct Edge{
int fr, to, next, v;
}e[2*maxn], e1[2*maxm];
bool cmp(const Edge &a, const Edge &b){ return a.v>b.v; }
int cnte, fir[maxn];
void addedge(int x, int y, int v){
Edge &ed=e[++cnte];
ed.to=y; ed.next=fir[x]; ed.v=v; fir[x]=cnte;
} int fa[maxn];
int find(int x){ return fa[x]==x?x:fa[x]=find(fa[x]); } int f[maxn][17], v[maxn][17]; //f表示不包括自己,上面第2^i个结点,v表示从i开始这个路径的权值
int dep[maxn];
void dfs(int now, int par, int val){ //当前结点 父亲结点 父亲连边的值
f[now][0]=par; v[now][0]=val; dep[now]=dep[par]+1;
for (int i=1; i<17; ++i){
f[now][i]=f[f[now][i-1]][i-1];
v[now][i]=min(v[now][i-1], v[f[now][i-1]][i-1]);
}
for (int i=fir[now]; i; i=e[i].next){
if (e[i].to==par) continue;
dfs(e[i].to, now, e[i].v);
}
} int n, m, q; int solve(int x, int y){
if (dep[x]<dep[y]) swap(x, y); int ans=INF;
for (int i=16; i>=0; --i)
if (dep[f[x][i]]>=dep[y]) ans=min(ans, v[x][i]), x=f[x][i];
for (int i=16; i>=0; --i)
if (f[x][i]!=f[y][i]) ans=min(ans, min(v[x][i], v[y][i])), x=f[x][i], y=f[y][i];
if (x!=y) ans=min(ans, min(v[x][0], v[y][0])); //一定要注意两个点是祖孙关系的情况!
return ans;
} int main(){
scanf("%d%d", &n, &m); int t1, t2, t3;
for (int i=0; i<m; ++i){
scanf("%d%d%d", &t1, &t2, &t3);
e1[i].fr=t1; e1[i].to=t2; e1[i].v=t3;
}
sort(e1, e1+m, cmp);
for (int i=1; i<=n; ++i) fa[i]=i;
for (int i=0; i<m; ++i){ //直接把边都跑一遍就行了!
if (find(e1[i].fr)==find(e1[i].to)) continue;
addedge(e1[i].fr, e1[i].to, e1[i].v);
addedge(e1[i].to, e1[i].fr, e1[i].v);
fa[find(e1[i].fr)]=find(e1[i].to);
}
for (int i=1; i<=n; ++i) if (fa[i]==i) dfs(i, 0, 0); //分成许多个子树
scanf("%d", &q);
for (int i=0; i<q; ++i){
scanf("%d%d", &t1, &t2);
if (find(t1)!=find(t2)){ puts("-1"); continue; }
printf("%d\n", solve(t1, t2));
}
return 0;
}

最新文章

  1. Javascript原型继承 __proto__
  2. Activity之间传递参数(三)
  3. [Storm] 内部消息缓存
  4. 结束日期必须大于开始日期--My97DatePicker日历控制的又一方便之处
  5. Oracle笔记 七、PL/SQL 异常处理
  6. What is the difference between Views and Materialized Views in Oracle?
  7. hdu 3033 I love sneakers!
  8. gcc 编译器
  9. 将 jsp 页面的值 传到struts2 action中(不是表单中的值)
  10. libevent总结学习
  11. ng-checked选择和点击增加dom
  12. Nexus 私有仓库搭建与 Maven 集成
  13. 豌豆夹Redis解决方案Codis源码剖析:Proxy代理
  14. s21day14 python笔记
  15. 第二章 FFmpeg常用命令
  16. android:onClick都做了什么
  17. 前端菜鸟起飞之学会ps切图
  18. 7 python 类的组合
  19. 【BZOJ3387】[Usaco2004 Dec]Fence Obstacle Course栅栏行动 线段树
  20. 通过API方式查看Azure Sign-ins记录

热门文章

  1. binlog之一:binary log初探
  2. HDU 2544 最短路(邻接表+优先队列+dijstra优化模版)
  3. 组装恢复rbd
  4. 刷题常用的STL容器总结
  5. [iOS]UIImageView增加圆角
  6. Windows Backdoor Tips
  7. Repmat:Replicate and tile an array
  8. dubbo错误排查之No provider available for the service
  9. 理解和正确使用Java中的断言(assert)
  10. maven ...../.m2/settings.xml