poj3164(最小树形图&朱刘算法模板)
题目链接:http://poj.org/problem?id=3164
题意:第一行为n, m,接下来n行为n个点的二维坐标, 再接下来m行每行输入两个数u, v,表点u到点v是单向可达的,求这个有向图的最小生成树即求最小树形图;
思路: 这是一道最小树形图模板题;
我们可以用朱刘算法来解:
朱刘算法只有3步,然后不断循环。
1:找到每个点的最小入边。既然是生成树,那么对于每个点来说,只要选一个权值最小的入边就可以了。
贪心思想。因为如果不是最小入边,那么它肯定不是最小树形图的一条边,考虑它是没有意义的。
2:找环。找环找的是最小入边构成的新图的环。如果没找到环,那么一棵树就已经形成了,
因为树就是没有环的图。再因为边权都是最小的,因此此时最小树形图就已经出来了,停止循环。
3:如果第2步中找到了环,那么这个环就可以缩成一个点。然后构造新图,更新边权。更新边权的方法是:
假设某点u在该环上,并设这个环中指向u的边权是in[u],那么对于每条从u出发的边(u, i, w),在新图中连接(new, i, w)的边,其中new为新加的人工顶点;对于每条进入u的边(i, u, w),在新图中建立边(i, new, w-in[u])的边。之所以是w-in[u]的原因是如果选择了w,那么那个in[u]在树中就是多余的,完全可以删除,所以需要减去,然后再后面的总费用累加中会体现出删掉了这个权值,不理解的画个图就明白了。
代码:
#include <iostream>
#include <stdio.h>
#include <math.h>
#include <string.h>
#include <algorithm>
#include <iomanip>
#define MAXN 210
using namespace std; const int inf=0x3f3f3f3f;
struct node{
int u, v;
double w; //**u到v的单向距离
}edge[MAXN*]; struct Point{ //**记录每个点的坐标
double x, y;
}point[MAXN]; double in_dist[MAXN]; //记录i点的最短入边距离
int pre[MAXN]; //**记录i节点的前继节点
int id[MAXN]; //**记录新图点节点
int vis[MAXN]; //**判环
int k=; //k为非自环边的数目 double dist(Point a, Point b){
return sqrt((a.x-b.x)*(a.x-b.x)+(a.y-b.y)*(a.y-b.y));
} void add(int u, int v, double dist){ //***加边
edge[k].u=u;
edge[k].v=v;
edge[k++].w=dist;
} double zhuliu(int root, int n){
double ans=;
while(true){
for(int i=; i<=n; i++){
in_dist[i]=inf;
}
for(int i=; i<k; i++){ //***找点i的最小入边
int u=edge[i].u;
int v=edge[i].v;
double w=edge[i].w;
if(u!=v&&in_dist[v]>w){//**注意这里新图也可能出现自环的情况,若不加u!=v的话会tle
in_dist[v]=w;
pre[v]=u; //**记录v的前继节点
}
}
for(int i=; i<=n; i++){
if(i==root){
continue;
}
ans+=in_dist[i];
if(in_dist[i]>=inf){
return -; //***根节点外还有孤立点,不存在最小树形图
}
}
memset(id, -, sizeof(id));
memset(vis, -, sizeof(vis));
int cnt=;
for(int i=; i<=n; i++){//***枚举每个点,找环
int v=i;
while(v!=root&&vis[v]!=i&&id[v]==-){//**上溯父节点找环
vis[v]=i;
v=pre[v];
}
if(v!=root&&id[v]==-){ //***找到了环,缩点并且从新编号
++cnt;
id[v]=cnt;
for(int u=pre[v]; u!=v; u=pre[u]){
id[u]=cnt;
}
}
}
if(!cnt){ //***没有出现环
break;
}
for(int i=; i<=n; i++){//***将余下的不在环中的点也重新编号
if(id[i]==-){
id[i]=++cnt;
}
}
for(int i=; i<k; i++){//***建新图
int u=edge[i].u;
int v=edge[i].v;
edge[i].u=id[u];
edge[i].v=id[v];
if(edge[i].u!=edge[i].v){
edge[i].w-=in_dist[v];
}
}
n=cnt; //**更新节点数目
root=id[root]; //**更新根节点编号
}
return ans;
} int main(void){
// freopen("test.in", "r", stdin);
// freopen("test.out", "w", stdout);
int u, v, n, m;
while(~scanf("%d%d", &n, &m)){
k=;
for(int i=; i<=n; i++){
scanf("%lf%lf", &point[i].x, &point[i].y);
}
while(m--){
scanf("%d%d", &u, &v);
if(u!=v){ //***删除自环
add(u, v, dist(point[u], point[v]));
}
}
int root=; //**本题中没有指定根节点,任取一个即可
double ans=zhuliu(root, n);
if(ans==-){
puts("poor snoopy");
}else{
printf("%.2f\n", ans); //**poj输出用lf会wa╭(╯^╰)╮
}
}
return ;
}
poj3164(最小树形图&朱刘算法模板)的更多相关文章
- 最小树形图——朱刘算法(Edmonds)
定义:一个有向图,存在从某个点为根的,可以到达所有点的一个最小生成树,则它就是最小树形图. 朱刘算法实现过程: [在选出入边集后(看步骤1),若有向图中不存在有向环,说明该图就是最小树形图] 1,选入 ...
- 洛谷P4716 【模板】最小树形图(朱刘算法)
题意 题目链接 Sol 朱刘算法?感觉又是一种神仙贪心算法 大概就是每次贪心的用每个点边权最小的入边更新答案,如果不行的话就缩起来找其他的边 不详细说了,丢链接走人.. #include<bit ...
- POJ 3164 Command Network ( 最小树形图 朱刘算法)
题目链接 Description After a long lasting war on words, a war on arms finally breaks out between littlek ...
- POJ 3164 Command Network 最小树形图 朱刘算法
=============== 分割线之下摘自Sasuke_SCUT的blog============= 最 小树形图,就是给有向带权图中指定一个特殊的点root,求一棵以root为根的有向生成树T, ...
- 最小树形图--朱刘算法([JSOI2008]小店购物)
题面 luogu Sol 首先设一个 \(0\) 号点,向所有点连边,表示初始价值 显然这个图的一个 \(0\) 为根的最小有向生成树的边权和就是每个买一次的最小价值 再买就一定能优惠(包含 \(0\ ...
- POJ - 3164-Command Network 最小树形图——朱刘算法
POJ - 3164 题意: 一个有向图,存在从某个点为根的,可以到达所有点的一个最小生成树,则它就是最小树形图. 题目就是求这个最小的树形图. 参考资料:https://blog.csdn.net/ ...
- bzoj 4349 最小树形图——朱刘算法
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4349. 学习博客:http://www.cnblogs.com/xzxl/p/7243466 ...
- UVA-11183 Teen Girl Squad (最小树形图、朱刘算法模板)
题目大意:给一张无向图,求出最小树形图. 题目分析:套朱-刘算法模板就行了... 代码如下: # include<iostream> # include<cstdio> # i ...
- POJ--3164--Command Network【朱刘算法】最小树形图
链接:http://poj.org/problem?id=3164 题意:告诉n个点坐标,m条边表示两个点之间有路.从1点開始建立一个有向图最小生成树. 朱刘算法模板题 =============== ...
随机推荐
- Android之Handler使用方法总结
方法一:(java习惯,在android平台开发时这样是不行的,由于它违背了单线程模型) 刚刚開始接触android线程编程的时候,习惯好像java一样,试图用以下的代码解决这个问题 new T ...
- ZOJ - 4016 Mergeable Stack 【LIST】
题目链接 http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=4016 题意 模拟栈的三种操作 第一种 push 将指定元素压入指 ...
- STM32 FSMC学习笔记+补充(LCD的FSMC配置)
STM32 FSMC学习笔记+补充(LCD的FSMC配置) STM32 FSMC学习笔记 STM32 FSMC的用法--LCD
- P5111 zhtobu3232的线段树
P5111 zhtobu3232的线段树 维护左子树右子树的贡献和跨区间贡献 #include<bits/stdc++.h> using namespace std; typedef lo ...
- uboot中添加自定义命令
uboot中可以通过修改源程序来添加自定义命令,进一步扩展uboot的功能. 我想在uboot下添加一条新的命令(名为varcpy),用来拷贝uboot中的环境变量. 修改方式如下: 创建新文件com ...
- rbx1包里机器人仿真程序的实践
git clone https://github.com/pirobot/rbx1.git 1.打开一个终端 cd ~/catkin_ws/ catkin_make source ./devel/s ...
- 思科安全:加密流量威胁检测、加密流量威胁和恶意软件检测、识别无线干扰或威胁、Talos 情报源可加强对已知和新型威胁的防御、分布式安全异常检测
思科DNA竞品比较工具 您的网络能够驱动数字化转型吗? 根据IDC调查,45%的受调研公司计划在未来两年内做好网络数字化的准备.查看数字化网络带来的结果和商业价值. 下载报告 思科 HPE 华为 Ar ...
- stack_2.由两个栈组成队列
思路: 用两个栈($stack_a, $stack_b),当push的时候,压入$stack_a, 让pop的时候,先把$stack_a元素依次全部倒入$stack_b中,再对$stack_b进行po ...
- Chapter2 二分与三分
T1 给一个N个数的序列,分成M段,每段最大值最小 sol:二分最大值,贪心Check T2 平面上n个点,每个点每s会向周围扩散一个单位长度,两个点联通当且仅当扩散有交点,问什么时候这n个点联通 s ...
- java反射中Class对象详解和类名.class, class.forName(), getClass()区别
1.获得类型类 可以看到,对象a是A的一个实例,A是某一个类,在if语句中使用a.getClass()返回的结果正是类A的类型类,在Java中表示一个特定类型的类型类可以用“类型.class”的方式获 ...