此题是最基础的最小生成树的题目,有两种方法, 一个是prim一个是kruskal算法,前者利用邻接矩阵,后者是利用边集数组

prim算法的思想是:一个点一个点的找, 先找从第一个点到其他点最小的, 把权值存放到一个lowcost的数组中,然后继续找下一个点,然后更新lowcost数组,注意,这时的lowcost不完全是第二个点到所有点的距离,而是,其他点到最小生成树的距离,然后一步一步的求,知道求完所有点

kruskal算法的思想是:先把边集数组按照权值进行排序,之后从最小的往上找,这时有个前提,就是不能有环,这样就能保证最大连通量为1,借助father数组

代码如下:

方法一(Prim):

 #include <stdio.h>
#include <string.h>
#include <algorithm>
using namespace std;
const int MAX = + ;
const int INFINITY = ;
int map[MAX][MAX];
int Sum;
int v, e;//v来存点的个数,e来存边的个数
void mini_span_tree_prim()
{
int lowcost[MAX];
lowcost[] = ;//标记已经加到最小生成树中
for(int i = ; i <= v; i++)
{
lowcost[i] = map[][i];
}
for(int i = ; i <= v; i++)
{
int j = ;
int index = ;
int minweight = INFINITY;
while(j <= v)
{//找出最小的边来,并保存其坐标
if(lowcost[j] != && lowcost[j] < minweight)
{
minweight = lowcost[j];
index = j;
}
j++;
}
lowcost[index] = ;//标记已经加到最小生成树中
Sum += minweight;
for(j = ; j <= v; j++)
{//判断没有加到最小生成树中的和比当前权值要小的点
if(lowcost[j] != && lowcost[j] > map[index][j])
{
lowcost[j] = map[index][j];
}
}
}
}
int main()
{
//freopen("1.txt", "r", stdin);
int n;
scanf("%d", &n);
while(n--)
{
Sum = ;
scanf("%d %d", &v, &e);
int t1, t2, t3;
memset(map, , sizeof(map));
for(int i = ; i <= v; i++)
{
map[i][i] = ;
}
for(int i= ; i < e; i++)
{
scanf("%d %d %d", &t1, &t2, &t3);
map[t1][t2] = t3;
map[t2][t1] = t3;
}
int mincost = INFINITY;
for(int i = ; i < v; i++)
{
scanf("%d", &t1);
mincost = min(mincost, t1);
}
mini_span_tree_prim();
printf("%d\n", Sum + mincost);
}
return ;
}

方法二(Kruskal):

#include <stdio.h>
#include <algorithm>
#include <cstring>
using namespace std;
typedef struct Node{
int s;
int e;
int weight;
}Node;
const int MAX = + ;
const int INFINITY = ;
Node arr[MAX * ];
int father[MAX];
int sum;
int v, edges; bool cmp(Node a, Node b)
{
return a.weight < b.weight;
} int find(int f)
{
while(father[f] > )
f = father[f];
return f;
}
//生成最小生成树
void mini_span_tree_kruskai()
{
int n, m;
int cnt = ;
for(int i= ; i < edges; i++)
{
n = find(arr[i].s);
m = find(arr[i].e);
if(cnt == v - )//优化,如果找到了n-1条边,这时退出就行了
break;
if(n != m)//判断是否构成了环
{
father[m] = n;
sum += arr[i].weight;
cnt++;
}
}
} int main()
{
int n;
scanf("%d", &n);
while(n--)
{
memset(arr, , sizeof(arr));
memset(father, , sizeof(father));
sum = ;
scanf("%d %d", &v, &edges);
for(int i = ; i < edges; ++i)
{
scanf("%d %d %d", &arr[i].s, &arr[i].e, &arr[i].weight);
}
int mincost = INFINITY;
int t;
for(int i = ; i <= v; i++)
{
scanf("%d", &t);
mincost = min(mincost, t);
}
sort(arr, arr + edges, cmp);//排序
// for(int i = 0; i < edges; i++)
// printf("%d %d %d\n", arr[i].s, arr[i].e, arr[i].weight);
// printf("%d\n", mincost);
mini_span_tree_kruskai();
printf("%d\n", sum + mincost); }
return ;
}

最小生成树--->NYOJ-38 布线问题的更多相关文章

  1. NYOJ 38布线问题

    http://acm.nyist.net/JudgeOnline/problem.php?pid=38 布线问题 时间限制:1000 ms  |  内存限制:65535 KB 难度:4   描述 南阳 ...

  2. NYOJ 38 布线问题 (最小生成树 prim)

    题目链接 描述 南阳理工学院要进行用电线路改造,现在校长要求设计师设计出一种布线方式,该布线方式需要满足以下条件: 1.把所有的楼都供上电. 2.所用电线花费最少 输入 第一行是一个整数n表示有n组测 ...

  3. nyoj 38 布线问题

    题目链接:http://acm.nyist.net/JudgeOnline/problem.php?pid=38 最小生成树水题~ 代码: #include "stdio.h" / ...

  4. NYOJ 38 布线问题_(解法2 Prim算法)

    时间限制:1000 ms  |  内存限制:65535 KB 难度:4 描写叙述 南阳理工学院要进行用电线路改造.如今校长要求设计师设计出一种布线方式.该布线方式须要满足下面条件: 1.把全部的楼都供 ...

  5. nyoj 38 布线问题 Kruskal and Prim

    布线问题 时间限制:1000 ms  |  内存限制:65535 KB 难度:4 描述 南阳理工学院要进行用电线路改造,现在校长要求设计师设计出一种布线方式,该布线方式需要满足以下条件: 1.把所有的 ...

  6. NYOJ 38 布线问题_(解法1 Kruskal算法)

    时间限制:1000 ms  |  内存限制:65535 KB 难度:4 描写叙述 南阳理工学院要进行用电线路改造.如今校长要求设计师设计出一种布线方式,该布线方式须要满足下面条件: 1.把全部的楼都供 ...

  7. 【待修改】nyoj 38 最小生成树

    package nyoj; import java.util.Scanner; public class Main { public static void main(String args[]) { ...

  8. nyoj 38 简单并查集的应用&最小生成树

    #include<stdio.h> #include<stdlib.h> #define inf 0x3fffffff #define N 600 struct node { ...

  9. NYIST OJ 题目38 布线问题

    最小生成树水题,先按最小生成树做,答案最后加上最小的从第i号楼接线到外界供电设施所需要的费用即可. #include<cstdio> #include<cstring> #in ...

  10. 最小生成树之kruskal方法实现 (java)

    今天是个阴天,下了点雨,work ......... 步骤:将所有边排序,然后不断从小到大加上边,这个过程最重要的是避免环的产生,此处用并查集.(nyoj 38) package 最小生成树; imp ...

随机推荐

  1. Extjs嵌入html

    方式一:使用组件的html属性嵌入html代码,如果html代码中存在参数可以使用字符串拼接的方式拼接html代码. html页面: <!doctype html> <html> ...

  2. 扩展《C程序设计语言》练习2-3程序通用性

    最近开始自学C语言,在看K&R的<C程序设计语言>.练习2-3要求写一个函数,将输入的十六进制数字字符串转换成与之等价的整数值,配套答案没有扩展程序的通用性,所以我就稍微改造改造. ...

  3. 栈的讲解 和 栈的生长方向 源代码技巧分析,简直没SEI 啦

    函数的局部变量,都是存放在"栈"里面,栈的英文是:STACK.STACK的大小,我们可以在stm32的启动文件里面设置,以战舰stm32开发板为例,在startup_stm32f1 ...

  4. 转:postgresql:pg_restore: [archiver] input file does not appear to be a valid archive的解决方法

      使用ps_restore恢复备份数据库出错:pg_restore: [archiver] input file does not appear to be a valid archive 使用pg ...

  5. Installshield更新时,新加dll未拷贝至安装目录问题完美解决【原创】

    最近在发现在开发过程中新增的目录或文件(特别是dll\ocx\exe等二进制文件),在升级安装时拷贝失败. 经过一周的研究,完美解决办法: 对于新增的文件或目录,放到一个新的component中,并且 ...

  6. cf B. Inna and Nine

    http://codeforces.com/contest/374/problem/B #include <cstdio> #include <cstring> #includ ...

  7. poj 2513Colored Sticks

    http://poj.org/problem?id=2513 #include<cstdio> #include<cstdlib> #include<cstring> ...

  8. PCB模擬設計接地的指導原則

    接地無疑是系統設計中最為棘手的問題之一.盡管它的概念相對比較簡單,實施起來卻很復雜,遺憾的是,它沒有一個簡明扼要可以用詳細步驟描述的方法來保證取得良好效果,但如果在某些細節上處理不當,可能會導致令人頭 ...

  9. 重新定义malloc和free 防止内存泄漏

    1, 定义供应用程序使用的头文件//libmem.h#ifndef _LIBMEM_H_#define _LIBMEM_H_ //声明自定义malloc及free函数extern void *my_m ...

  10. 彻底卸载 RAD Studio 2009/2010/XE+ 的步骤

    重新安装 RAD 系列时,建议将上一个版本彻底卸载,彻底卸载 RAD Studio 2009/2010/XE+ 的步骤: 控制面板-->添加/删除程序中执行了卸载操作以后, 还需要做以下工作: ...