题目大意:给出N个点。M条边。问这N个点形成的生成树的最大权值边-最小权值边的最小值

解题思路:先排序,然后按生成树的kruscal算法进行加边,再维护一个最小权值边

加边的时候要考虑一下加下去的边是否会形成环,假设形成环的话,就把环内的最小边去掉,然后再找出这棵新的生成树的最小边

等到生成树形成的时候,由于加入进去的新边的权值肯定是最大值的,所以仅仅要仅仅减去之前维护一个的最小值就能够了

#include <cstdio>
#include <cstring>
#include <algorithm>
#include <vector>
using namespace std; #define N 400
#define M 160010
#define INF 0x3f3f3f3f struct Edge{
int u, v, c;
Edge() {}
Edge(int u, int v, int c): u(u), v(v), c(c) {}
}E[M]; int f[N], dis[N][N];
int cnt, Min_Edge, n, m;
bool vis[N]; int cmp(const Edge &a, const Edge &b) {
return a.c < b.c;
} void init() {
for (int i = 0; i < m; i++) {
scanf("%d%d%d", &E[i].u, &E[i].v, &E[i].c);
dis[E[i].u][E[i].v] = dis[E[i].v][E[i].u] = E[i].c;
}
} int LCA(int i) {
int u = E[i].u, v = E[i].v, c = E[i].c;
memset(vis, 0, sizeof(vis));
while (!vis[u]) {
vis[u] = true;
if (u == f[u]) break;
u = f[u];
} while (!vis[v] && f[v] != v) v = f[v];
if (!vis[v]) return -1;
return v;
} void findCycle(int i) {
int lca = LCA(i);
if (lca == -1) return ;
int u = E[i].u, v = E[i].v, c = E[i].c;
Edge MinEdge;
MinEdge.c = INF;
int fu = f[u];
while (fu != u && u != lca) {
if (dis[fu][u] < MinEdge.c) MinEdge = Edge(fu, u, dis[fu][u]); fu = f[fu];
u = f[u]; } int fv = f[v];
while (fv != v && v != lca) {
if (dis[fv][v] < MinEdge.c) MinEdge = Edge(fv, v, dis[fv][v]);
fv = f[fv];
v = f[v];
} f[MinEdge.v] = MinEdge.v;
Min_Edge = INF;
for (int i = 0; i < n; i++)
if (f[i] != i && dis[f[i]][i] < Min_Edge)
Min_Edge = dis[f[i]][i];
cnt--;
} void AddEdge(int i) {
int u = E[i].u, v = E[i].v, c = E[i].c;
if (f[u] == u) f[u] = v;
else if (f[v] == v) f[v] = u;
else {
vector<int> vec;
while (1) {
vec.push_back(u);
if (u == f[u]) break;
u = f[u];
}
int size = vec.size();
for (int i = size - 1; i > 0; i--) f[vec[i]] = vec[i - 1];
f[E[i].u] = E[i].v;
}
Min_Edge = min(Min_Edge, c);
cnt++;
} void solve() {
sort(E, E + m, cmp);
for (int i = 0; i < n; i++)
f[i] = i; int ans = INF;
Min_Edge = INF;
cnt = 0;
for (int i = 0; i < m; i++) {
findCycle(i);
AddEdge(i);
if (cnt == n - 1) ans = min(ans, E[i].c - Min_Edge); }
printf("%d\n", ans);
} int main() {
while (scanf("%d", &n) != EOF && n) {
scanf("%d", &m);
init();
solve();
}
return 0;
}

UVALive - 4960 Sensor network(生成树+LCA)的更多相关文章

  1. uvalive 4960 Sensor Network

    题意: 给出一个无向图,求一个生成树使得这个生成树的最大边与最小边之差最小,输出这个最小的差值.n的最大值为350. 思路: 这题不看题解想破头也不知道怎么写Orz. 暴力的做法是可以从大到小枚举边作 ...

  2. 【暑假】[实用数据结构]UVAlive 3027 Corporative Network

    UVAlive 3027 Corporative Network 题目:   Corporative Network Time Limit: 3000MS   Memory Limit: 30000K ...

  3. POJ 1459 Power Network / HIT 1228 Power Network / UVAlive 2760 Power Network / ZOJ 1734 Power Network / FZU 1161 (网络流,最大流)

    POJ 1459 Power Network / HIT 1228 Power Network / UVAlive 2760 Power Network / ZOJ 1734 Power Networ ...

  4. 【BZOJ4080】【WF2014】Sensor Network [随机化]

    Sensor Network Time Limit: 2 Sec  Memory Limit: 128 MB[Submit][Status][Discuss] Description 魔法炮来到了帝都 ...

  5. poj 3417 Network(tarjan lca)

    poj 3417 Network(tarjan lca) 先给出一棵无根树,然后下面再给出m条边,把这m条边连上,然后每次你能毁掉两条边,规定一条是树边,一条是新边,问有多少种方案能使树断裂. 我们设 ...

  6. 【Uvalive4960】 Sensor network (苗条树,进化版)

    [题意] 给出N个点,M条边,问这N个点形成的生成树的最大权值边-最小权值边的最小值 InputThe input consists of several test cases, separated ...

  7. bzoj3322 最大生成树+LCA

    题目大意:给个无向图,每条边有个限制,每个点最多能买入和卖出一定黄金:然后按顺序走过n个点,求每个卖出黄金的点最多能卖出多少黄金 一开始有点懵,想着怎么再图上做这个问题,后来知道要先建一棵最大生成树 ...

  8. ACM-ICPC 2018 徐州赛区网络预赛 J Maze Designer(最大生成树+LCA)

    https://nanti.jisuanke.com/t/31462 题意 一个N*M的矩形,每个格点到其邻近点的边有其权值,需要构建出一个迷宫,使得构建迷宫的边权之和最小,之后Q次查询,每次给出两点 ...

  9. 最大生成树——LCA

    今天说是要练习LCA结果找了道题看着题解打完了,如此惭愧,Lca还得好好理解啊,感觉在最大生成树上做有点异样,可能还是不是很理解吧,在noip前一定要再把这道题再a一遍,好题啊. 这是2013noip ...

随机推荐

  1. wscript shell

    http://blog.csdn.net/songques/article/details/8309569 http://baike.baidu.com/link?url=_P6z73_Ih9R79T ...

  2. UVALive 6527 Counting ones dfs(水

    题目链接:点击打开链接 #include <cstdio> #include <vector> using namespace std; typedef long long l ...

  3. 推断一个java文件和邮箱格式是否合法

    import java.util.Scanner; public class StringTest { public static void main(String[] args) { int bac ...

  4. [NOI.AC#33]bst 线段树

    链接 区间修改,完全二叉树,这引导我们把这棵树看成一棵线段树 .线段树的每一个节点相当于这棵二叉树的节点, 对于区间交换操作,我们对二叉树的每一层从上到下分别考虑,找到L,R在第i层对应的节点修改 这 ...

  5. 中间件 —— 消息中间件(MOM)

    维基百科对消息中间件的定义为:Message-oriented middleware (MOM) is software or hardware infrastructure supporting s ...

  6. Linux shell command学习笔记(一)

    Shell的种类有很多种,例如CSH,Bourne Shell,Korn Shell.在现在的大多数Linux发行版中,默认的Shell一般都是Bourne again shell(bash). &l ...

  7. 数组-reduce方法

    转自: https://github.com/Advanced-Frontend/Daily-Interview-Question/issues/139 实现 convert 方法,把原始 list ...

  8. 自绘listCtrl控件选中该行高亮(模拟windows)

    CListCtrl的派生类CMyListCtrl的DrawItem()函数里添加代码 CDC *pDC = CDC::FromHandle(lpDrawItemStruct->hDC); if ...

  9. 设计模式六大原则(二):里氏替换原则(Liskov Substitution Principle)

    里氏替换原则(LSP)由来: 最早是在 妖久八八 年, 由麻神理工学院得一个女士所提出来的. 定义: 1:如果对每一个类型为 T1的对象 o1,都有类型为 T2 的对象o2,使得以 T1定义的所有程序 ...

  10. 洛谷 P1927 防护伞

    P1927 防护伞 题目描述 据说 2012 的灾难和太阳黑子的爆发有关.于是地球防卫小队决定制造一个特殊防护 伞,挡住太阳黑子爆发的区域,减少其对地球的影响.由于太阳相对于地球来说实在是太 大了,我 ...