kuangbin专题八 POJ3164 Command Network(最小树形图的理解+模板)

本文主要是介绍kuangbin专题八 POJ3164 Command Network(最小树形图的理解+模板),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

题意:
给你N个点的坐标,然后给你M条单向边的信息,A,B表示A指向B,从1开始,1的信息能传达到所有的点。
题解:
一开始这道题我看到了单向边,还想着怎么把他变成双向边来做prim算法,后来实在想不到,去看了题解,发现这道题原来是一个最小树形图。。。ORZ,对于这个算法的理解我还是看了别的大佬的博客才弄懂的,看算法的话可以参考一下这个大佬的:http://blog.csdn.net/wsniyufang/article/details/6747392
然后要想看的通俗点的话就看一下这个大佬的博客:
http://blog.csdn.net/sdj222555/article/details/7459738

再之后是我自己对于上面两位大佬的讲解进行的理解:

(朱刘算法)最小树形图是求有向边的最小生成树的算法。
算法步骤如下:
1.判断图的连通性,若不连通直接无解,否则一定有解。
2.为除了根节点以外的所有点选择一个权值最小的入边,假设用pre数组记录前驱,f数组记录选择的边长,记所选边权和为temp。
3.(可利用并查集)判断选择的的边是否构成环,若没有则直接ans+=temp并输出ans,若有,则进行下一步操作。
4.对该环实施缩点操作,设该环上有点V1,V2……Vi……Vn,缩成的点为node?,对于所有不在环中的点P进行如下更改:
(1)点P到node的距离为min{a[p,Vi]-f[Vi]} (a为边集数组)
(2)点node到p的距离为min{a[Vi,p]}
操作(1)的理解:先假设环上所有边均选上,若下次选择某一条边进入该环,则可以断开进入点与进入点的前驱之间的边,即断开F[进入点],所以等效为直接把a[p,node]赋值为min{a[p,Vi]-f[Vi]}。

算法中的a[p,vi]的意思是点p指向vi的长度,第四步中(1)通俗的理解的话可以这样想:
举个例子:某个图的部分图中, 1->2权值为3, 2->1权值为4, 3->1权值为9, 4->2权值为7。那么可以看到,结点1和结点2是形成了一个环的。
我们仅从其大小是不知道该删除哪条边比较好(这个就好像无向图联通三个点,1<->3权值为100,1<->2权值为1,2<->3权值为5,一开始从1走,判断2与3,你能马上确定不要1<->3这条边吗?),
这时看到3->1权值为9,如果走这条边,那么接下来只能删除掉2->1这条边,同理走4->2的话就要删除掉1->2这条边。那么就不妨建立新图, 将1和2缩成一点,3->1的权值就变成了9-4=5, 4->2的权值变成了7-3=4。这样的话,
就相当于变相删除了不需要走的边了(我来说一下我的理解,1和2缩点之后,3->1的权值为什么要9-4=5,因为你要在有向图中联通4个点的话,3连到1,而你不能要2->1,而应该要1->2的边的吧?因为你要2->1的话就会出现3能通1不能通2的尴尬情况,这对于要联通整个图是不合理的,到这里懂了吧?同理4->2的权值变成7-3=4也应该能理解了吧?如果选择1->2的话就会出现4通2不通1的情况了。所以减去那些入边的操作实际上是为了删除不需要的边)。
形成新图后,又变成了最小树形图的求解,就这样循环下去,直到图中的最小边集没有环为止。

//大佬的模板,ORZ
#include<stdio.h>
#include<string.h>
#include<math.h>
#include<algorithm>
using namespace std;
#define INF 0x3f3f3f3f
const int MAXN=105;
struct node1
{int u,v;double w;
}map[MAXN*MAXN];
struct node2
{double x,y;
}p[MAXN];
double IN[MAXN];
int pre[MAXN];
int ID[MAXN];
int vis[MAXN];
int n,m;
double zhuliu_mst(int root,int V,int E)
{memset(pre,0,sizeof(pre));double sum=0;while(true){//1.找最小入边for(int i=1;i<=V;i++) IN[i]=INF*1.0;for(int i=1;i<=E;i++){int u=map[i].u;int v=map[i].v;if(map[i].w<IN[v]&&u!=v){pre[v]=u;IN[v]=map[i].w;}}for(int i=1;i<=V;i++){if(i==root) continue;if(IN[i]==INF*1.0) return -1;//除了根以外的点没有入边,则根无法到达该点 }//2找环int cnt=1;memset(ID,0,sizeof(ID));memset(vis,0,sizeof(vis));IN[root]=0;for(int i=1;i<=V;i++)//标记每个环 {sum+=IN[i];int v=i;while(vis[v]!=i&&!ID[v]&&v!=root){vis[v]=i;v=pre[v];}if(v!=root&&!ID[v]){for(int u=pre[v];u!=v;u=pre[u]){ID[u]=cnt;}ID[v]=cnt++; } }if(cnt==1) break;//无环for(int i=1;i<=V;i++){if(!ID[i])ID[i]=cnt++;}//3.缩点,重新标记for(int i=1;i<=E;i++){int v=map[i].v;int u=map[i].u;map[i].u=ID[u];map[i].v=ID[v];if(ID[u]!=ID[v]){map[i].w-=IN[v];}}V=cnt-1;//为什么要-1,因为是从1开始标记的,i++之后会多出1,所以要减去1 root=ID[root];}return sum;
}
int main()
{while(~scanf("%d%d",&n,&m)){for(int i=1;i<=n;i++)scanf("%lf%lf",&p[i].x,&p[i].y);for(int i=1;i<=m;i++){scanf("%d%d",&map[i].u,&map[i].v);if(map[i].u!=map[i].v)map[i].w=sqrt((p[map[i].u].x-p[map[i].v].x)*(p[map[i].u].x-p[map[i].v].x)+(p[map[i].u].y-p[map[i].v].y)*(p[map[i].u].y-p[map[i].v].y));elsemap[i].w=INF;}double ans=zhuliu_mst(1,n,m);if(ans==-1)printf("poor snoopy\n");elseprintf("%.2f\n",ans);}return 0;
}

这篇关于kuangbin专题八 POJ3164 Command Network(最小树形图的理解+模板)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/670117

相关文章

认识、理解、分类——acm之搜索

普通搜索方法有两种:1、广度优先搜索;2、深度优先搜索; 更多搜索方法: 3、双向广度优先搜索; 4、启发式搜索(包括A*算法等); 搜索通常会用到的知识点:状态压缩(位压缩,利用hash思想压缩)。

poj3468(线段树成段更新模板题)

题意:包括两个操作:1、将[a.b]上的数字加上v;2、查询区间[a,b]上的和 下面的介绍是下解题思路: 首先介绍  lazy-tag思想:用一个变量记录每一个线段树节点的变化值,当这部分线段的一致性被破坏我们就将这个变化值传递给子区间,大大增加了线段树的效率。 比如现在需要对[a,b]区间值进行加c操作,那么就从根节点[1,n]开始调用update函数进行操作,如果刚好执行到一个子节点,

C++11第三弹:lambda表达式 | 新的类功能 | 模板的可变参数

🌈个人主页: 南桥几晴秋 🌈C++专栏: 南桥谈C++ 🌈C语言专栏: C语言学习系列 🌈Linux学习专栏: 南桥谈Linux 🌈数据结构学习专栏: 数据结构杂谈 🌈数据库学习专栏: 南桥谈MySQL 🌈Qt学习专栏: 南桥谈Qt 🌈菜鸡代码练习: 练习随想记录 🌈git学习: 南桥谈Git 🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈�

【专题】2024飞行汽车技术全景报告合集PDF分享(附原数据表)

原文链接: https://tecdat.cn/?p=37628 6月16日,小鹏汇天旅航者X2在北京大兴国际机场临空经济区完成首飞,这也是小鹏汇天的产品在京津冀地区进行的首次飞行。小鹏汇天方面还表示,公司准备量产,并计划今年四季度开启预售小鹏汇天分体式飞行汽车,探索分体式飞行汽车城际通勤。阅读原文,获取专题报告合集全文,解锁文末271份飞行汽车相关行业研究报告。 据悉,业内人士对飞行汽车行业

poj 1258 Agri-Net(最小生成树模板代码)

感觉用这题来当模板更适合。 题意就是给你邻接矩阵求最小生成树啦。~ prim代码:效率很高。172k...0ms。 #include<stdio.h>#include<algorithm>using namespace std;const int MaxN = 101;const int INF = 0x3f3f3f3f;int g[MaxN][MaxN];int n

poj 1287 Networking(prim or kruscal最小生成树)

题意给你点与点间距离,求最小生成树。 注意点是,两点之间可能有不同的路,输入的时候选择最小的,和之前有道最短路WA的题目类似。 prim代码: #include<stdio.h>const int MaxN = 51;const int INF = 0x3f3f3f3f;int g[MaxN][MaxN];int P;int prim(){bool vis[MaxN];

poj 2349 Arctic Network uva 10369(prim or kruscal最小生成树)

题目很麻烦,因为不熟悉最小生成树的算法调试了好久。 感觉网上的题目解释都没说得很清楚,不适合新手。自己写一个。 题意:给你点的坐标,然后两点间可以有两种方式来通信:第一种是卫星通信,第二种是无线电通信。 卫星通信:任何两个有卫星频道的点间都可以直接建立连接,与点间的距离无关; 无线电通信:两个点之间的距离不能超过D,无线电收发器的功率越大,D越大,越昂贵。 计算无线电收发器D

poj 1734 (floyd求最小环并打印路径)

题意: 求图中的一个最小环,并打印路径。 解析: ans 保存最小环长度。 一直wa,最后终于找到原因,inf开太大爆掉了。。。 虽然0x3f3f3f3f用memset好用,但是还是有局限性。 代码: #include <iostream>#include <cstdio>#include <cstdlib>#include <algorithm>#incl

hdu 1102 uva 10397(最小生成树prim)

hdu 1102: 题意: 给一个邻接矩阵,给一些村庄间已经修的路,问最小生成树。 解析: 把已经修的路的权值改为0,套个prim()。 注意prim 最外层循坏为n-1。 代码: #include <iostream>#include <cstdio>#include <cstdlib>#include <algorithm>#include <cstri

【生成模型系列(初级)】嵌入(Embedding)方程——自然语言处理的数学灵魂【通俗理解】

【通俗理解】嵌入(Embedding)方程——自然语言处理的数学灵魂 关键词提炼 #嵌入方程 #自然语言处理 #词向量 #机器学习 #神经网络 #向量空间模型 #Siri #Google翻译 #AlexNet 第一节:嵌入方程的类比与核心概念【尽可能通俗】 嵌入方程可以被看作是自然语言处理中的“翻译机”,它将文本中的单词或短语转换成计算机能够理解的数学形式,即向量。 正如翻译机将一种语言