nyoj_118:修路方案(次小生成树)
题意,判断次小生成树与最小生成树的权值和是否相等。
豆丁文档—— A-star和第k短路和次小生成树和Yen和MPS寻路算法
法一:
先求一次最小生成树,将这棵树上的边加入一个向量中,再判断去掉前面所求的最小生成树的某条边能否再求得一棵等权值的最小生成树
复杂度O(NElogE)
//稠密图不可用
#include<bits/stdc++.h>
using namespace std;
+ ;
+ ;
int fa[maxn];
int n, m;
vector<int> TreeEdge;
struct Road //保存每条路的信息
{
int u,v,w;
bool operator<(const Road& t) const //按长度由小到大排序
{
return w<t.w;
}
} R[maxm];
//初始化并查集
void Init()
{
; i <= n; i++)
fa[i]=i; //每个点自成一个连通分量
}
//找x的祖先
int Find(int x)
{
return x == fa[x]? x:fa[x]=Find(fa[x]);
}
//判断图的连通性
bool isAccess()
{
);
; i <= n; i++)
if(f!=Find(i)) return false;
return true;
}
//kruskal求不要k这条边的最小生成树
int kruskal(int k)
{
;
Init();
; i<m; i++)
{
if(i!=k)
{
int tx=Find(R[i].u);
int ty=Find(R[i].v);
if(tx!=ty)
{
ans+=R[i].w;
fa[tx]=ty;
}
}
}
if(isAccess()) return ans; //如果是生成树,返回权值
;
}
int main()
{
int t;
scanf("%d", &t);
while (t--)
{
TreeEdge.clear();
scanf("%d%d",&n,&m);
; i<m; i++)
scanf("%d%d%d",&R[i].u,&R[i].v,&R[i].w);
sort(R, R+m); //将所有边排序
Init();
//===begin===MST
; //求最小生成树
; i<m; i++)
{
int tx=Find(R[i].u), ty=Find(R[i].v);
if(tx!=ty)
{
ans+=R[i].w;
fa[tx]=ty;
TreeEdge.push_back(i);//将这条边加入生成树中
}
}
//===end===MST
;
;i<TreeEdge.size();i++)
if(kruskal(TreeEdge[i])==ans) //看看不用这条边能否再次生成一棵最小生成树
{
ok=;
break;
}
if(ok) printf("Yes\n");
else printf("No\n");
}
}
法二:
首先求出原图的最小生成树,记录权值之和为Minst.枚举添加每条不在最小生成树上的边<u,v>,加上以后一定会形成一个环,找到环上权值第二大的边(即除<u,v>外最大的边)把它删除掉,计算当前生成树的权值之和。取所有枚举修改的生成树权值之和的最小值,就是次小生成树。具体实现时,更简单的方法是从每个节点i遍历整个最小生成树,定义F[i,j]为从i到j的路径上最大边的权值。遍历图求出F[i,j]的值,然后对于添加每条不在最小生成树中的边<i,j>,新的生成树权值之和就是Minst+w<i,j>-F[j],记录其最小值,则为次小生成树。该算法的时间复杂度为O(n^2+m)。由于只用求一次最小生成树,可以用最简单的Prim算法,时间复杂度为O(n^2)。算法的瓶颈不在于最小生成树,而在于O(n^2+m)的枚举加边修改,所以用更好的最小生成树算法是没有必要的。(详见豆丁文档)
复杂度O(n^2)
标程如下:
#include<bits/stdc++.h>
using namespace std;
#define maxN 510
#define MAX 0x0fffffff
#define MIN -0x0fffffff
int N,M,map[maxN][maxN],dis[maxN],maxlen[maxN][maxN],pre[maxN];
bool vis[maxN];
int prim()
{
int i,j,k,minn,pr;
memset(vis,false,sizeof(vis));
; i<=N; i++)
dis[i]=map[][i],pre[i]=;
vis[]=true;
; j<N; j++)
{
minn=MAX;
; i<=N; i++)
if(!vis[i]&&dis[i]<minn)
k=i,minn=dis[i];
pr=pre[k];
maxlen[k][pr]=maxlen[pr][k]=map[k][pr];
; i<=N; i++)
if(vis[i])
maxlen[i][k]=maxlen[k][i]=max(maxlen[i][pr],maxlen[k][pr]);
vis[k]=true;
; i<=N; i++)
if(!vis[i]&&dis[i]>map[i][k])
{
dis[i]=map[i][k];
pre[i]=k;
}
}
; i<N; i++)
; j<=N; j++)
if(pre[i]==j||pre[j]==i)continue;
;
;
}
int main()
{
int T;scanf("%d",&T);
while(T--)
{
int u,v,w;
scanf("%d%d",&N,&M);
; i<=N; i++)
; j<=N; j++)
{
map[i][j]=MAX;
maxlen[i][j]=MIN;
}
; i<M; i++)
{
scanf("%d%d%d",&u,&v,&w);
map[u][v]=map[v][u]=w;
}
if(prim())printf("Yes\n");
else printf("No\n");
}
}
nyoj_118:修路方案(次小生成树)的更多相关文章
- Nyoj 修路方案(次小生成树)
描述 南将军率领着许多部队,它们分别驻扎在N个不同的城市里,这些城市分别编号1~N,由于交通不太便利,南将军准备修路. 现在已经知道哪些城市之间可以修路,如果修路,花费是多少. 现在,军师小工已经找到 ...
- hdu4081 秦始皇修路(次小生成树)
题目ID:hdu4081 秦始皇修路 题目链接:点击打开链接 题目大意:给你若干个坐标,每个坐标表示一个城市,每个城市有若干个人,现在要修路,即建一个生成树,然后有一个魔法师可以免费造路(不消耗人 ...
- 修路方案 Kruskal 之 次小生成树
次小生成树 : Kruskal 是先求出来 最小生成树 , 并且记录下来所用到的的边 , 然后再求每次都 去掉最小生成树中的一个边 , 这样求最小生成树 , 然后看能不能得到 和原来最小生成树一样的 ...
- nyoj--118--修路方案(次小生成树)
修路方案 时间限制:3000 ms | 内存限制:65535 KB 难度:5 描述 南将军率领着许多部队,它们分别驻扎在N个不同的城市里,这些城市分别编号1~N,由于交通不太便利,南将军准备修路. ...
- 修路方案(nyoj)
算法:次小生成树 描述 南将军率领着许多部队,它们分别驻扎在N个不同的城市里,这些城市分别编号1~N,由于交通不太便利,南将军准备修路. 现在已经知道哪些城市之间可以修路,如果修路,花费是多少. 现在 ...
- NYOJ 118 修路方案
修路方案 时间限制:3000 ms | 内存限制:65535 KB 难度:5 描述 南将军率领着许多部队,它们分别驻扎在N个不同的城市里,这些城市分别编号1~N,由于交通不太便利,南将军准备修 ...
- hdu4081 次小生成树变形
pid=4081">http://acm.hdu.edu.cn/showproblem.php?pid=4081 Problem Description During the Warr ...
- HDU 4081 Qin Shi Huang's National Road System 次小生成树变种
Qin Shi Huang's National Road System Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/3 ...
- 【次小生成树】bzoj1977 [BeiJing2010组队]次小生成树 Tree
Description 小 C 最近学了很多最小生成树的算法,Prim 算法.Kurskal 算法.消圈算法等等. 正当小 C 洋洋得意之时,小 P 又来泼小 C 冷水了.小 P 说,让小 C 求出一 ...
随机推荐
- 每天一个Linux命令—— crontab
一.简介 1. Linux中有许多周期性执行的任务,这些任务由cron守护进程完成,因此cron这个系统服务是默认启动的. 2.Linux中调度任务分为两类:系统任务调度和用户任务调度,系统任务调度即 ...
- JAVAEE学习——struts2_01:简介、搭建、架构、配置、action类详解和练习:客户列表
一.struts2是什么 1.概念 2.struts2使用优势以及历史 二.搭建struts2框架 1.导包 (解压缩)struts2-blank.war就会看到 2.书写Action类 public ...
- Redis 集群搭建详细指南
先有鸡还是先有蛋? 最近有朋友问了一个问题,说毕业后去大城市还是小城市?去大公司还是小公司?我的回答都是大城市!大公司! 为什么这么说呢,你想一下,无论女孩男孩找朋友都喜欢找个子高胸大的.同样的道理嘛 ...
- Javascript中的数组去重-indexof方法
在Javascript中,有时我们会用到数组去重.我在这里给大家介绍一下本人认为最简单实用的一种方法-indexOf()去重. var arr = [1,1,1,2,2,2,3,3,4,5,6,2,1 ...
- Windows 7安装Oracle 10g的方法
Windows7下安装Oracle 10g提示"程序异常终止,发生未知错误"的解决方法 1.修改Oracle 10G\database\stage\prereq\db\refhos ...
- Hibernate_Validator学习分享
1. Hibernate Validator介绍 1.1 背景 在任何时候,当你要处理一个应用程序的业务逻辑,数据校验是你必须要考虑和面对的事情.应用程序必须通过某种手段来确保输入进来的数据从 ...
- Kafka官方文档翻译——简介
简介 Kafka擅长于做什么? 它被用于两大类应用: 在应用间构建实时的数据流通道 构建传输或处理数据流的实时流式应用 几个概念: Kafka以集群模式运行在1或多台服务器上 Kafka以topics ...
- ZooKeeper源码分析-Jute-第二部分
数据类型和流 本部分描述支持Hadoop的基础数据类型以及复合类型.我们的目的是支持一系列的类型,可以用于在不同的编程语言中简化和有效表达一定范围的记录类型. 基础类型 大部分情况下,Hadoop的大 ...
- nodejs+websocket制作聊天室视频教程
本套教程主要讲解了node平台的安装,node初级知识.node 服务器端程序响应http请求,通过npm安装第三方包,websocket即时通讯.聊天页面界面制作.拖动原理.拖动效果.遮罩效果.定位 ...
- linux开机启动smb服务
修改/etc/rc.local文件(增加红色部分) [root@localhost ~]# cat /etc/rc.local #!/bin/sh## This script will be exec ...