题意:

给出一个无向图,求一个生成树使得这个生成树的最大边与最小边之差最小,输出这个最小的差值。n的最大值为350。

思路:

这题不看题解想破头也不知道怎么写Orz。

暴力的做法是可以从大到小枚举边作为最小边的权值,求MST,但是复杂度达到了O(n^4),很显然会T。

考虑在kruskal算法加边的时候,当两个点在同一个连通分量的时候,加入这条边会形成环,这个时候就把环中的最小边去掉,剩下的边就尽可能达到了最大,当前加入的边设为Ea也最大的,然后再找现在生成树中的最小边Eb;当有n-1条边的时候,就形成了一个生成树,就可以利用Ea – Eb去更新答案。

最重要的两个过程是找环和加边。

判断环,用了LCA的思想,但并不是倍增,而是纯粹暴力的dfs。对于一条边的两个点假设为x和y,首先从一个点x开始访问,直到访问到它的祖先,即满足条件par[x] == x,标记这个过程中的所有点;然后从y开始访问,直到访问到它的祖先(同上)或者它自己或某一个祖先被标记了为止,若y访问到祖先都没有点被标记,那么就说明x和y没有公共祖先,那么它们就不在同一个连通分量中,加入这条边就不会形成环。

若找到了环,那么就要求这个环中边的最小值,求法很自然,遍历从x出发到lca的所有边,遍历从y出发到lca的所有边,找出其中的最小边,之后去掉这条边,去掉这条边有个技巧,假设最小边的非父亲点为u,那么去掉这条边时,令par[u] = u,这条边就被去掉了。之后再找最小边即可。

之后就是加边的过程。并查集加边是启发式合并,但是这题的加边是有方向的加边

假设两个点x和y,如果有其中一个点就是祖先,那么就把这个点连到另外一个点上(如x为祖先点,那么令par[x] = y)。

但是如果两个点都不是祖先,那么就要考虑如何合并,因为一个点不可能同时有两个父亲

可以考虑从一个点开始访问,直到访问到祖先,将路径上的点全部按访问的顺序记录在一个vector中,之后逆向加边。

再把这个点连到另一个点上。如图:

合并前:

合并后:

最后再把这个点(从这个点开始访问的)连接到另一个点上,加边就完成了。

枚举边的复杂度为O(n^2),找环和加边的复杂度为O(n),所以总的复杂度为O(n^3)。

感谢MZjj帮我debug!

代码:

 #include <stdio.h>
#include <string.h>
#include <iostream>
#include <algorithm>
#include <vector>
using namespace std; const int N = ;
const int inf = 0x3f3f3f3f; struct edge
{
int x,y,w; edge(){}; edge(int a,int b,int c)
{
x = a;
y = b;
w = c;
}
}; vector<edge> es; int par[N];
int n,m;
int dis[N][N];
bool vis[N];
int num;
int Minedge;
struct edge minedge; bool cmp(edge aa,edge bb)
{
return aa.w < bb.w;
} void init(void)
{
for (int i = ;i < n;i++) par[i] = i; num = ; es.clear(); memset(dis,inf,sizeof(dis)); Minedge = inf;
} int LCA(int x,int y)
{
memset(vis,,sizeof(vis)); while ()
{
vis[x] = ; if (x == par[x]) break; x = par[x];
} while (!vis[y] && y != par[y])
{
y = par[y];
} if (!vis[y]) return -; return y;
} void findcycle(int k)
{
int x = es[k].x,y = es[k].y; int lca = LCA(x,y); if (lca == -) return; minedge.w = inf; while (x != par[x] && x != lca)
{
if (dis[x][par[x]] < minedge.w)
{
minedge = edge(par[x],x,dis[x][par[x]]);
} x = par[x];
} while (y != par[y] && y != lca)
{
if (dis[y][par[y]] < minedge.w)
{
minedge = edge(par[y],y,dis[y][par[y]]);
} y = par[y];
} par[minedge.y] = minedge.y; Minedge = inf; for (int i = ;i < n;i++)
{
Minedge = min(dis[par[i]][i],Minedge);
} num--;
} void adde(int k)
{
int x = es[k].x,y = es[k].y; if (x == par[x]) par[x] = y;
else if (y == par[y]) par[y] = x;
else
{
vector<int> v; while ()
{
v.push_back(x); if (x == par[x]) break; x = par[x];
} for (int i = v.size() - ;i > ;i--) par[v[i]] = v[i-]; par[es[k].x] = es[k].y;
} num++; Minedge = min(Minedge,es[k].w);
} int main()
{
while (scanf("%d",&n) && n)
{
scanf("%d",&m); init(); for (int i = ;i < m;i++)
{
int a,b,c; scanf("%d%d%d",&a,&b,&c); es.push_back(edge(a,b,c));
} for (int i = ;i < m;i++)
{
int x = es[i].x,y = es[i].y; dis[x][y] = dis[y][x] = es[i].w;
} sort(es.begin(),es.end(),cmp); int ans = inf; for (int i = ;i < m;i++)
{
findcycle(i);
adde(i); if (num == n - )
{
//cout << es[i].w << " ** " << Minedge << endl;
ans = min(ans,es[i].w - Minedge);
}
} cout << ans << endl;
} return ;
}

uvalive 4960 Sensor Network的更多相关文章

  1. UVALive - 4960 Sensor network(生成树+LCA)

    题目大意:给出N个点.M条边.问这N个点形成的生成树的最大权值边-最小权值边的最小值 解题思路:先排序,然后按生成树的kruscal算法进行加边,再维护一个最小权值边 加边的时候要考虑一下加下去的边是 ...

  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. 【Uvalive4960】 Sensor network (苗条树,进化版)

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

  6. UVALive 3027 Corporative Network

    ---恢复内容开始--- Corporative Network Time Limit: 3000MS   Memory Limit: Unknown   64bit IO Format: %lld ...

  7. 泛在传感器网络(Ubiquitous Sensor Network; USN)

    http://wiki.mbalib.com/wiki/%E6%B3%9B%E5%9C%A8%E4%BC%A0%E6%84%9F%E5%99%A8%E7%BD%91%E7%BB%9C 什么是泛在传感器 ...

  8. UVALive 3027 Corporative Network 带权并查集

                         Corporative Network A very big corporation is developing its corporative networ ...

  9. 并查集 + 路径压缩(经典) UVALive 3027 Corporative Network

    Corporative Network Problem's Link Mean: 有n个结点,一开始所有结点都是相互独立的,有两种操作: I u v:把v设为u的父节点,edge(u,v)的距离为ab ...

随机推荐

  1. idea设置文件的编码格式为utf-8

    file ---> settings... 然后 editor --->  file encoding 然后设置path和encoding什么的.

  2. Python开发【笔记】:asyncio 定时器

    asyncio 定时器 实现: import asyncio class Timer: def __init__(self, timeout, callback): self._timeout = t ...

  3. 如何实现一个字符的反转 (Java)

    一..通过jdk自带reverse的方法.

  4. 使用Git,如何忽略不需要上传的文件(配置文件)

    步骤1:在目录下,选择GIt Bash Here 2.输入命令 : git update-index --assume-unchanged 文件名 3.再输入指令 git  status 查看修改文件 ...

  5. 运行vs2010,Debug时发生“无法启动程序"http://localhost:xxx",系统找不到指定文件

    网上参考回答:应该是Visual Studio不能启动浏览器导致的结果!第一种结果是没有设置默认的浏览器,第二种情况是注册表被修改了! 我的尝试: 1.在启动处点击下三角,选择“使用以下浏览器浏览” ...

  6. java Map 怎么遍历

    java中遍历MAP的几种方法 Java代码 Map<String,String> map=new HashMap<String,String>();    map.put(& ...

  7. [MySQL 5.6] information_schema.innodb_metrics

    1. 概括 已关闭/打开的配置 use information_schema select count(*), status from innodb_metrics group by status; ...

  8. GPT转MBR怎么转?

    GPT转MBR分区怎么转?现在很多笔记本的硬盘分区都是GPT模式,如果想装XP的话,那只能将GPT磁盘转换成MBR磁盘分区才行.接下来,简单说说如何将GPT分区转成MBR分区! 如果本身电脑有两个硬盘 ...

  9. rpm: /root/anaconda3/lib/liblzma.so.5: version `XZ_5.1.2alpha' not found (required by /lib64/librpmio.so.3)

    报如上的错误,发现rpm相关的命令都不能够用. 1.搜到这篇文章,https://stackoverflow.com/questions/47633870/rpm-lib64-liblzma-so-5 ...

  10. Hadoop生态集群之HDFS

    一.HDFS是什么 HDFS是hadoop集群中的一个分布式的我文件存储系统.他将多台集群组建成一个集群,进行海量数据的存储.为超大数据集的应用处理带来了很多便利. 和其他的分布式文件存储系统相比他有 ...