POJ 1679 The Unique 次最小生成树 MST
http://poj.org/problem?id=1679
题目大意:
给你一些点,判断MST(最小生成树)是否唯一。
思路:
以前做过这题,不过写的是O(n^3)的,今天学了一招O(n^2)的,哈哈~
方法一:
首先先建立MST,然后把这个MST的边一个个尝试不使用,构建另外一颗MST,然后判断权值是否相等。
这样复杂度需要O(n^3)。。
方法二:
还可以用次最小生成树的方法解决:如果最小生成树不唯一,那么次小生成树的权值和最小生成树相同。
我们可以枚举要加入哪一条新边。在最小生成树上加一条边u-v之后,图上会出现一条回路。因此删除的边必须在最小生成树u到v的路径上,而且是这条路径上的最长边。而次最小生成树一定能由最小生成树加一条再删除一条边得到。只需我们求出没对结点u和v在最小生成树中唯一路径的最大边权dp[i][j],剩下的只需O(m)时间,枚举所有的m-n+1条边进行交换,每次花O(1)时间求出新生成树的权值。总时间复杂度为O(n^2)
方法一:
#include<cstdio>
#include<algorithm>
using namespace std;
const int MAXN=101;
int fa[MAXN]; struct point
{
int x,y;
int len;
}data[MAXN*MAXN],pre[MAXN*MAXN]; //pre 记录等一下用到了哪些条边 bool operator < (const point &a ,const point &b) //sort 重载比较函数
{
return a.len<b.len;
} void UFinit(int n) //并查集初始化
{
for(int i=1;i<=n;i++)
fa[i]=i;
} int find(int cur) //带路径压缩的并查集查询函数,简洁而优雅~
{
return fa[cur]==cur? cur : fa[cur]=find(fa[cur]);
} int kruskal(int n,int m,int &prelen)
{
UFinit(n);
int ans=0;
for(int i=0;i<m;i++)
{
int rootx=find(data[i].x);
int rooty=find(data[i].y);
if(rootx!=rooty)
{
fa[rootx]=rooty;
ans+=data[i].len;
pre[prelen++]=data[i]; }
} return ans;
} int kruskal2(int n,int m,int nox,int noy)//nox noy 代表直接连接这两个点之间的边不选
{
UFinit(n); int ans=0;
for(int i=0;i<m;i++)
{
if(data[i].x==nox && data[i].y==noy)
continue; int rootx=find(data[i].x);
int rooty=find(data[i].y);
if(rootx!=rooty)
{
fa[rootx]=rooty;
ans+=data[i].len;
}
} return ans; }
int main()
{ int T;
scanf("%d",&T);
while(T--)
{
int n,m;
scanf("%d%d",&n,&m);
for(int i=0;i<m;i++)
scanf("%d%d%d",&data[i].x,&data[i].y,&data[i].len); int prelen=0; //pre数组的长度
sort(data,data+m); //kruskal 前面的工作 int ans=kruskal(n,m,prelen); //正常版本的kruskal bool unique=true; for(int i=0;i<prelen;i++)
{
int res=kruskal2(n,m,pre[i].x,pre[i].y);//带去除边的kruskal int cnt=0;
for(int j=1;j<=n;j++)
if(fa[i]==i)
cnt++; if(cnt!=0) //这个很重要!没有就WA,因为可能剩下的不连通,但是恰好res==ans
continue; if(res==ans) //如果还有一棵生成树和第一次所求的权值一样,说明最小生成树不唯一
{
unique=false;
break;
}
} if(unique)
printf("%d\n",ans);
else
printf("Not Unique!\n");
}
return 0;
}
方法二:
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
const int MAXN=100+10;
int n,m,head[MAXN],len,fa[MAXN],nodes[MAXN],n_len;
int dp[MAXN][MAXN];
bool vis[MAXN];
int find(int cur) //带路径压缩的并查集查询函数,简洁而优雅~
{
return fa[cur]==cur? cur : fa[cur]=find(fa[cur]);
} struct edge //MST邻接表
{
int to,next;
double val;
}e[MAXN*MAXN]; void add(int from,int to,double val)
{
e[len].to=to;
e[len].val=val;
e[len].next=head[from];
head[from]=len++;
}
struct point
{
int x,y;
int len;
bool operator < (const point &b) const//sort 重载比较函数
{
return len<b.len;
} }data[MAXN*MAXN]; void dfs(int cur,int fa,int dis)
{
for(int i=0;i<n_len;i++)
{
int x=nodes[i];
dp[cur][x]=dp[x][cur]=max(dp[x][fa],dis);
} nodes[n_len++]=cur; for(int i=head[cur];i!=-1;i=e[i].next)
{
int id=e[i].to;
if(id != fa)
dfs(id,cur,e[i].val);
}
} int main()
{
int T;
scanf("%d",&T);
while(T--)
{
memset(head,-1,sizeof(head));
len=0;
memset(vis,0,sizeof(vis));
scanf("%d%d",&n,&m);
for(int i=0;i<m;i++)
scanf("%d%d%d",&data[i].x,&data[i].y,&data[i].len); for(int i=1;i<=n;i++)
fa[i]=i; //kruskal
sort(data,data+m);
int mstlen=0;
for(int i=0;i<m;i++)
{
int x=data[i].x,y=data[i].y;
int root_x=find(x),root_y=find(y);
if(root_x!=root_y)
{
mstlen+=data[i].len;
fa[root_x]=root_y;
add(x,y,data[i].len); //kruskal中顺便建立MST的邻接表
add(y,x,data[i].len);
vis[i]=true;
}
}
n_len=0;
memset(dp,0,sizeof(dp));
dfs(1,-1,0);
int mstlen2=9999999;
for(int i=0;i<m;i++)
{
if(vis[i]) continue;
int x=data[i].x,y=data[i].y,dis=data[i].len;
mstlen2=min(mstlen2,mstlen - dp[x][y] + dis);
}
// printf("%d\n",mstlen2);
if(mstlen == mstlen2)
puts("Not Unique!");
else
printf("%d\n",mstlen);
}
return 0;
}
POJ 1679 The Unique 次最小生成树 MST的更多相关文章
- poj 1679 The Unique MST 【次小生成树】【模板】
题目:poj 1679 The Unique MST 题意:给你一颗树,让你求最小生成树和次小生成树值是否相等. 分析:这个题目关键在于求解次小生成树. 方法是,依次枚举不在最小生成树上的边,然后加入 ...
- poj 1679 The Unique MST(唯一的最小生成树)
http://poj.org/problem?id=1679 The Unique MST Time Limit: 1000MS Memory Limit: 10000K Total Submis ...
- poj 1679 The Unique MST (判定最小生成树是否唯一)
题目链接:http://poj.org/problem?id=1679 The Unique MST Time Limit: 1000MS Memory Limit: 10000K Total S ...
- POJ 1679 The Unique MST 【最小生成树/次小生成树模板】
The Unique MST Time Limit: 1000MS Memory Limit: 10000K Total Submissions: 22668 Accepted: 8038 D ...
- poj 1679 The Unique MST
题目连接 http://poj.org/problem?id=1679 The Unique MST Description Given a connected undirected graph, t ...
- POJ 1679 The Unique MST (次小生成树kruskal算法)
The Unique MST 时间限制: 10 Sec 内存限制: 128 MB提交: 25 解决: 10[提交][状态][讨论版] 题目描述 Given a connected undirect ...
- POJ 1679 The Unique MST(判断最小生成树是否唯一)
题目链接: http://poj.org/problem?id=1679 Description Given a connected undirected graph, tell if its min ...
- 【POJ 1679 The Unique MST】最小生成树
无向连通图(无重边),判断最小生成树是否唯一,若唯一求边权和. 分析生成树的生成过程,只有一个圈内出现权值相同的边才会出现权值和相等但“异构”的生成树.(并不一定是最小生成树) 分析贪心策略求最小生成 ...
- POJ 1679 The Unique MST 【判断最小生成树是否唯一】
Description Given a connected undirected graph, tell if its minimum spanning tree is unique. Defini ...
随机推荐
- jQuery对表格进行类样式
<%-- <%@ page language="java" contentType="text/html; charset=utf-8" pageE ...
- 56.如何清除已经设置的npm config配置
npm config delete registry npm config delete disturl 或者 npm config edit 找到淘宝那两行,删除
- BZOJ 1355 KMP中next数组的应用
思路: 我们知道 next[i]是失配的i下一步要去哪儿 next[n]就是失配的n要去哪儿 n-next[n]就是答案(即最短周期)啦 //By SiriusRen #include <cst ...
- zabbix3.4.7搭建及邮件告警
Zabbix3.4.7部署 系统环境:CentOs7.2 1.关闭selinux 1.1 [root@localhost ~]# setenforce 0 #临时关闭 1.2 [root@localh ...
- .Net 程序在自定义位置查找托管/非托管 dll 的几种方法
原文:.Net 程序在自定义位置查找托管/非托管 dll 的几种方法 一.自定义托管 dll 程序集的查找位置 目前(.Net4.7)能用的有2种: #define DEFAULT_IMPLEMENT ...
- 【Codeforces Round #427 (Div. 2) B】The number on the board
[Link]:http://codeforces.com/contest/835 [Description] 原本有一个数字x,它的各个数码的和原本是>=k的; 现在这个数字x,在不改变位数的情 ...
- Android-Volley网络通信框架(二次封装数据请求和图片请求(包含处理请求队列和图片缓存))
1.回想 上篇 使用 Volley 的 JsonObjectRequest 和 ImageLoader 写了 电影列表的样例 2.重点 (1)封装Volley 内部 请求 类(请求队列,数据请求,图片 ...
- 软件project经验总结系列之三 - 计划阶段控制
本文为软件project经验总结系列文章的第三篇.按照总论文章所设立的范围划分,本阶段重点讲述计划阶段的控制过程以及控制思路,笔者所站在的角度是乙方角度来进行表述整个阶段的推动过程,但对于甲方公司其基 ...
- UVa 11094 - Continents
题目:有一些岛屿在湖中.地图用两种字符表示.当前处在位置是一个岛屿.求除了当前岛屿外的最大岛屿. 分析:图论,floodfill.直接利用dfs求联通部分的面积就可以,然后取出最大. 说明:横线没有边 ...
- java初始化过程中成员变量
package day01; class Base{ int j; //1.j=0 Base(){ add(1); //2.调用子类add()方法 System.out.println(j); //4 ...