- 浏览: 202003 次
- 性别:
- 来自: 武汉
文章分类
- 全部博客 (137)
- c++ (74)
- c++,算法,回溯 (2)
- DP问题。 (9)
- DP问题,0/1背包问题 (3)
- 数学问题 (6)
- 贪心算法 (10)
- 排序 (16)
- 数据结构 (7)
- 容器 (2)
- 模拟问题 (2)
- 水题 (8)
- 并查集 (3)
- 非技术 (2)
- 素数问题 (1)
- DFS (3)
- 二叉树 (1)
- 递归 (1)
- 图论 (5)
- 最小生成树 (5)
- 最短路径 (6)
- bell_flaod算法 (2)
- hash (3)
- 二分查找 (1)
- 搜索 (5)
- BFS (5)
- STL (3)
- 字符串hash (1)
- 拓扑排序 (1)
- 字典树 (4)
- 哈弗曼树 (1)
- KMP (7)
- 线段树 (9)
- 树状数组 (6)
- 全排列 (2)
- DP问题 (2)
- LCS (1)
- 最长不下降子序列 (2)
- 面试经验 (3)
kruskal思想:
对图G的所有边按权值从小到大排序,每次取一条边,若此边的两个结点属于同一个连通分量,则舍弃这条边;若不属于同一个连通分量,则将这条边加入到最小生成树中。当所有结点属于同一个连通分量时,构造完毕。
kruskal设计:
一个边的结构体。有几个关键问题:
如何判断两个结点是否属于一个连通分量:
每个连通分量都用其中一个结点来标识,p[]数组用来表示结点的父节点(初始p[i]=i),真正的头结点满足p[i]=i;首先根据传进来的边的两个顶点,用findSet找出点所属的父节点(递归,条件x==p[x]),若父节点相同,显然属于同一个连通分量,直接返回,测试下一条边;否则,合并两个连通分量。
如何合并两个连通分量:
两个连通分量x,y,是将x合并到y还是y合并到x?这就用一个rank数组来确定(初始rank[]=0)。如果rank[x]>rank[y],则将y合并到x,即p[y] = x;否则,将x合并到y,即p[x] = y,如果rank[x]=rank[y],还要将rank[y]加1。
例:
初始传入最小边ab,rank[a]==0==rank[b],则将rank[b]加1,p[a]=b;
传入边ac,a属于分量b,c属于分量c,且rank[b]==1 > 0==rank[c],则p[c]=b;
同理传入ad,ae之后,p[d],p[e]都等于b。
传入边fg,同传入ab;
传入边bg,rank[b]==1==rank[g],则p[g]=b,rank[b]++;
传入边be,b与e同属于b,直接返回,最小生成树构造完毕!
代码如下:
#include <iostream> using namespace std; int n; //结点个数 int m; //边的条数 int p[26]; //用来寻找某个点属于的子连通图(P[i] == i) int rank[26]; //若i属于某连通分量且p[i] = i;则用rank[i]来判断合并两个连通分量将哪个作为哪个的子集 int result; typedef struct edge { int v1; int v2; int w; }Edge; Edge e[75]; int cmp(const void *a, const void *b) { return (*(Edge *)a).w - (*(Edge *)b).w; } void initial() { memset(e,0,sizeof(Edge)*75); m = 0; result = 0; } int findSet(int x) //找属于的集合 { if(x != p[x]) p[x] = findSet(p[x]); return p[x]; } void Union(int x, int y, int w) /*****更改集合*****/ { if(x == y) return; if(rank[x] > rank[y]) { p[y] = x; } else { p[x] = y; if(rank[x] == rank[y]) rank[y] ++; } result += w; } int kruskal() { int i; for(i = 0; i < 26; i++) { rank[i] = 0; p[i] = i; } qsort(e,m,sizeof(Edge),cmp); for(i = 0; i < m; i++) Union(findSet(e[i].v1), findSet(e[i].v2), e[i].w); return result; } int main() { int i,j; char v1; int num; char v2; int w; while(cin>>n && n!=0) { initial(); for(i = 1; i < n; i++) { cin>>v1>>num; for(j = 0; j < num; j++) { cin>>v2>>w; e[m].v1 = v1-'A'; e[m].v2 = v2 -'A'; e[m].w = w; m++; } } printf("%d\n",kruskal()); } return 0; }
发表评论
-
虚函数、纯虚函数、虚基类、抽象类、虚函数继承、虚继承
2013-08-29 14:34 836虚函数:虚函数是C++中用于实现多态(polymorphis ... -
排序算法总结
2013-05-17 11:00 841选择排序、快速排序、希尔排序、堆排序不是稳定的排序算法, ... -
poj 3122
2012-12-11 19:51 866题意:作者要开一个生日party,他现在拥有n块高度都为1 ... -
poj 3273
2012-12-11 16:49 982题意:给你天数n,和每天需要花的钱,让你把这些天分成m份(每份 ... -
字典树学习材料
2012-05-30 14:29 967字典树,又称单词查找树,Trie树,是一种树形结构,典型应 ... -
poj 1159
2012-05-28 19:08 1441题目大意:给你一段字符串,让你求出在中间最少加入几个字符 ... -
poj 3176
2012-05-28 14:47 1016大致题意: 输入一个n层的三角形,第i层有i个数,求从第 ... -
poj 1260
2012-05-28 09:54 1610题意解释: 有n个等级的珠宝,等级依次升高,等级越高价钱越高 ... -
poj 1836
2012-05-28 09:22 2710是POJ2533的扩展题。题意不难,令到原队列 ... -
poj 2533
2012-05-26 15:36 1269在做这道题目之前,首先让我们了解一下什么是LIS算法,LIS俗 ... -
poj 3267
2012-05-26 09:43 801从程序可以看出,第i个位置到L所删除的字符数,总是先取最坏情况 ... -
poj 1276
2012-05-25 16:20 2392题意: 这道题的意思是给你一堆钱,各种面值的都有,比 ... -
poj 1094
2012-05-25 13:54 1107题意:给出字母个数,和有限个有序对(a<b)求出能确定字 ... -
poj 3393
2012-05-23 17:00 1256大致题意: 科普文一篇,文章80%都是无用信息,因为 ... -
poj 3007
2012-05-14 10:21 991大致题意: 给定一个字符串,从任意位置把它切为两半, ... -
poj 3096
2012-05-10 21:09 1003题意: 定义D-pairs表示取字符串s中相距为D的两个字母 ... -
poj 1426
2012-04-26 20:11 2170大致题意: 给出一个整数n,(1 <= n <= ... -
poj 1797
2012-04-24 15:05 1623题目大意是就是何处一个图,n个顶点和m条边,每个边都有最大承载 ... -
poj 1338
2012-04-23 10:20 1257题意:题目意思是求由2,3,5的乘积组成的数从大到小排列,从1 ... -
poj 2021
2012-04-19 15:00 948题意:Ted今年100岁,给出n对他家族的关系:“父 ...
相关推荐
在解答此题时,首先我们要对马的速度进行排序,然后按照贪心的原则,让速度最快的马去参加第一场比赛,第二快的马去参加第二场比赛,以此类推。这样可以确保在每一轮比赛中,我们都有可能获胜。当然,实际的比赛情况...
【标签】"3310 poj"进一步确认了这是关于POJ平台上的第3310号问题,通常这类问题会涉及到特定的算法或者编程挑战,可能是数据结构、图论、动态规划、搜索算法等领域的经典问题。 【压缩包子文件的文件名称列表】...
第二题是Aggressive cows(poj2456)。题目给出一系列坐标点,要求将牛放置在这些点上,使得任意两头牛之间的距离尽可能大。同样,这也是一个最大化最小值的问题,而且依旧可以采用二分查找配合验证的解法。通过对...
2. 状态转移:对于第i个裁判,有两种情况:包含该裁判和不包含该裁判。若包含,则需要找到与前i-1个裁判有共识的裁判子集;若不包含,则直接考虑前i-1个裁判。通过遍历所有可能的子集,更新dp[i][mask]的值。 3. ...
【描述】"西工大POJ习题第五季,数组这类的代码,有截图" 涉及的知识点: 1. **数组基础**:数组是最基本的数据结构之一,是连续存储同一类型元素的集合。在这些习题中,学生需要掌握一维、二维数组的创建、初始化...
输入包含两行,第一行是蛋糕的体积`N`(不超过10000),第二行是层数`M`(不超过20)。输出是一个正整数`S`,表示满足条件的最小面积,如果无解则输出`S = 0`。 题目中提供的代码是使用深度优先搜索(DFS)策略来...
"2.24.1"可能是一个内部版本号,表示这是作者对问题多次尝试后的第2.24.1个版本。这通常意味着在解决复杂问题时,开发者可能会经历多次迭代和优化,以找到最有效的解决方案。 【标签】代码 "代码"标签表明了这个...
2. POJ1716【基础】:这是POJ1201的一个简化版本,要求每个区间至少有两个点,解法与前者类似。 3. POJ3159【基础】和POJ3169【中等】:这两个题目可能涉及到更复杂的差分约束问题,如奶牛的排列问题,需要考虑不同...
第一行的粉刷方案有2^N种可能性,通过位压缩存储技巧减少运行时间和空间需求。 #### 时钟问题 时钟问题(POJ1166)要求用最少的移动次数,将3×3时钟矩阵的所有时钟指针拨至12点位置。该问题具有有限的可能性,共计...
北京大学-动态规划-讲解PDF的内容涉及了动态规划这一重要的算法设计范式,它是一种在计算机科学和数学优化领域内解决复杂问题的策略,尤其适用于有着大量重叠子问题和最优子结构特性的问题。接下来将详细阐述文档中...
- **解法:**通过遍历每个柱子,并利用单调栈找到每个柱子左右两侧第一个比它矮的柱子的位置,从而确定以该柱子为高的最大矩形的范围。 - **关键步骤:** 1. 对于每个柱子`x`,找到左侧第一个比`x`矮的柱子的位置。...