题目链接:uva 11354 - Bond

题目大意:给定一张图。每次询问两个节点路径上进过边的危急值的最大值的最小值。

解题思路:首先建立最小生成数,然后依据这棵树做树链剖分。

#include <cstdio>
#include <cstring>
#include <vector>
#include <algorithm> using namespace std;
const int maxn = 50005;
const int INF = 0x3f3f3f3f; struct Edge {
int u, v, w;
Edge (int u = 0, int v = 0, int w = 0) { set(u, v, w); }
void set(int u, int v, int w) {
this->u = u;
this->v = v;
this->w = w;
}
friend bool operator < (const Edge& a, const Edge& b) {
return a.w < b.w;
}
}ed[maxn * 2]; int N, M, Q, ne, f[maxn], first[maxn], jump[maxn * 2], val[maxn];
int id, idx[maxn], top[maxn], far[maxn], son[maxn], dep[maxn], cnt[maxn];
vector<Edge> vec; inline int getfar(int x) {
return x == f[x] ? x : f[x] = getfar(f[x]);
} inline void add_Edge (int u, int v, int w) {
ed[ne].set(u, v, w);
jump[ne] = first[u];
first[u] = ne++;
} void dfs (int u, int pre, int d) {
far[u] = pre;
dep[u] = d;
son[u] = 0;
cnt[u] = 1; for (int i = first[u]; i + 1; i = jump[i]) {
int v = ed[i].v;
if (v == pre)
continue;
dfs(v, u, d + 1);
cnt[u] += cnt[v];
if (cnt[son[u]] < cnt[v])
son[u] = v;
}
} void dfs (int u, int rot) {
top[u] = rot;
idx[u] = ++id;
if (son[u])
dfs(son[u], rot);
for (int i = first[u]; i + 1; i = jump[i]) {
int v = ed[i].v;
if (v == far[u] || v == son[u])
continue;
dfs(v, v);
}
} #define lson(x) ((x)<<1)
#define rson(x) (((x)<<1)|1)
int lc[maxn << 2], rc[maxn << 2], s[maxn << 2]; inline void pushup(int u) {
s[u] = max(s[lson(u)], s[rson(u)]);
} void build (int u, int l, int r) {
lc[u] = l;
rc[u] = r; if (l == r) {
s[u] = val[l];
return;
} int mid = (l + r) / 2;
build(lson(u), l, mid);
build(rson(u), mid + 1, r);
pushup(u);
} int query(int u, int l, int r) {
if (l <= lc[u] && rc[u] <= r)
return s[u];
int mid = (lc[u] + rc[u]) / 2, ret = 0;
if (l <= mid)
ret = max(ret, query(lson(u), l, r));
if (r > mid)
ret = max(ret, query(rson(u), l, r));
return ret;
} void init () {
int u, v, w;
ne = id = 0;
vec.clear();
memset(first, -1, sizeof(first));
for (int i = 1; i <= N; i++)
f[i] = i; for (int i = 0; i < M; i++) {
scanf("%d%d%d", &u, &v, &w);
vec.push_back(Edge(u, v, w));
}
sort(vec.begin(), vec.end());
for (int i = 0; i < vec.size(); i++) {
int p = getfar(vec[i].u);
int q = getfar(vec[i].v);
if (p == q)
continue;
add_Edge(vec[i].u, vec[i].v, vec[i].w);
add_Edge(vec[i].v, vec[i].u, vec[i].w);
f[p] = q;
} dfs(1, 0, 0);
dfs(1, 1);
for (int i = 0; i < N - 1; i++) {
int t = i * 2;
u = (dep[ed[t].u] < dep[ed[t].v] ? ed[t].v : ed[t].u);
val[idx[u]] = ed[t].w;
}
build(1, 1, N);
} int solve (int u, int v) {
int p = top[u], q = top[v], ret = 0;
while (p != q) {
if (dep[p] < dep[q]) {
swap(p, q);
swap(u, v);
}
ret = max(ret, query(1, idx[p], idx[u]));
u = far[p];
p = top[u];
}
if (u == v)
return ret;
if (dep[u] > dep[v])
swap(u, v);
ret = max(ret, query(1, idx[son[u]], idx[v]));
return ret;
} int main () {
int cas = 0;
while (scanf("%d%d", &N, &M) == 2) {
if (cas++)
printf("\n"); init(); int u, v;
scanf("%d", &Q);
while (Q--) {
scanf("%d%d", &u, &v);
printf("%d\n", solve(u, v));
}
}
return 0;
}

版权声明:本文博客原创文章。博客,未经同意,不得转载。

最新文章

  1. Visual Studio 默认保存为UTF8编码
  2. git 本地分支与远程分支
  3. 开发android过程中eclipse闪退解决
  4. 三、saltstack证书管理
  5. 2016MBA排名
  6. Winform合并多个Excel文件到一个文件中(源文件.xls,实际是.xml)
  7. Build Settings
  8. 在云服务器搭建WordPress博客(二)使用xampp并解决端口冲突问题
  9. 关于ionic的一些坑(2)
  10. 【c基础】之 文件及其操作
  11. zoj3785 What day is that day?
  12. 什么是MVC ?
  13. 自定义 js 文件的集成引用
  14. 在多机器上远程执行JMeter
  15. python构建bp神经网络_曲线拟合(一个隐藏层)__2.代码实现
  16. tensorflow 高级api使用分布式之配置
  17. as3 去掉字符串空白问题
  18. C# 使用Newtonsoft.Json序列化自定义类型
  19. Python开发【模块】:sqlalchemy
  20. git 常用的撤销操作

热门文章

  1. 返璞归真 asp.net mvc (3) - Controller/Action
  2. 乐在其中设计模式(C#) - 享元模式(Flyweight Pattern)
  3. oracle dblink造成远程数据库session过多
  4. ListView装上拉电阻下拉刷新
  5. Alamofire网络库基础教程
  6. CentOS-6.5-x86_64 最小化安装后,怎样安装 man 程序?
  7. 于SharePoint经营SharePoint Designer建立
  8. 好大滴坑, Spring MVC覆盖了Trsaction
  9. [ACM] HDU 2295 Radar (二分法+DLX 重复覆盖)
  10. ViewGroup可实现上下、各地跑马灯效果滚动