思路:

  比较典型的最小生成树的题目了、、在这里用求最小生成树的经典算法K(Kruskal)算法和P(Prim)算法。我的 K 算法用的是结构体来存图,P 算法用的是邻接矩阵来存图,K算法的复杂度是O(ElogE),比较适用于稀疏图,P算法的复杂的是O(V ^ 2),适合用稠密图。

以下是C++的K算法代码

 #include <iostream>
#include <cstdio>
#include <cstring>
#include <cstdlib>
#include <cmath>
#include <cctype>
#include <algorithm>
using namespace std; const int MAXN = 2e3+ ;
int pre[MAXN];
int m,n; int Find(int x) //并查集
{
return x == pre[x] ? x :(pre[x] = Find(pre[x]));
} struct Node //存图
{
int u, v, w;
}cy[]; int mycmp(Node a,Node b)
{
return a.w < b.w;
} void mst()
{
for(int i = ; i < ; i++)
pre[i] = i;
} int kru()//算法的具体实现
{
int ans = ;
int cnt = ;
for(int i = ; i <= n; i++)
{
int fv = Find(cy[i].v);
int fu = Find(cy[i].u);
if(fv != fu)
{
pre[fv] = fu;
ans += cy[i].w;
cnt ++;
}
if(cnt == m -)
{
return ans;
break;
}
}
return -;
} int main()
{
//freopen("in.cpp","r",stdin);
while(~scanf("%d%d",&n,&m) && n)
{
mst();
for(int i = ; i <= n; i++)
scanf("%d%d%d",&cy[i].u, &cy[i].v, &cy[i].w);
sort(cy + , cy + n + , mycmp);
int ans = kru();
if(ans != -)
printf("%d\n",ans);
else
printf("?\n");
}
return ;
}

以下是C++的P算法代码

 #include <iostream>
#include <cstdio>
#include <cstring>
#include <cstdlib>
#include <cmath>
#include <cctype>
#include <algorithm>
using namespace std; const int MAXN = ;
const int INF = 0x3f3f3f3f;
int edge[MAXN][MAXN]; //用于存放图
int used[MAXN];//用于标记点是否已加入到最小生成树那个的集合中
int lowcost[MAXN]; //用于存放集合外的点到集合内的点的最短距离,每加入一个点需要更新一次
int N,M; int prim(int start,int maxn) //利用prim算法计算最小生成树
{
memset(used, , sizeof(used));
for(int i = ; i <= maxn; i++)
{
lowcost[i] = edge[start][i];
}
used[start] = ;
int sumweight = ;
int ok = ;
for(int i = ; i <= maxn; i++)
{
int minn = INF ;
int v = -;
for(int j = ; j <= maxn; j++)
{
if(used[j] == && lowcost[j] < minn)
{
minn = lowcost[j];
v = j;
}
}
//cout << v << " ";
if(v != -)
{
ok++;
used[v] = ;
sumweight += lowcost[v];
for(int j = ; j <= maxn; j++)
{
if(used[j] == && lowcost[j] > edge[v][j])
{
lowcost[j] = edge[v][j];
}
}
}
}
if(ok == maxn -)
return sumweight;
return -;
} int main()
{
while(cin >> N >> M && N)
{
memset(edge, 0x3f, sizeof(edge));
while(N--)
{
int u, v, w;
cin >> u >> v >> w;
edge[u][v] = edge[v][u] = w;
}
int ans = prim(, M);
if(ans < ) cout << "?" <<endl;
else cout << ans << endl;
}
return ;
}

以下是JAVA的K算法

 import java.util.Scanner;
import java.util.Comparator;
import java.util.Arrays;
import java.text.DecimalFormat; class Vge{
int u;
int v;
int w;
} class mycmp implements Comparator<Vge>{
public int compare( Vge A, Vge B ){
if( A.w - B.w != 0 )
return A.w - B.w;
else
return A.w - B.w;
}
} public class Main{
final static int MAXN = 100 + 3;
static int[] pre = new int[ MAXN ];
static Vge[] clg = new Vge[ MAXN * MAXN ];
public static void main( String[] args ){
Scanner sc = new Scanner( System.in );
int n, m;
while( sc.hasNextInt() ){
n = sc.nextInt();
m = sc.nextInt();
if( n == 0 ) break;
for( int i = 0; i < n; i++ ){
clg[ i ] = new Vge();
clg[ i ].u = sc.nextInt();
clg[ i ].v = sc.nextInt();
clg[ i ].w = sc.nextInt();
}
mst( m );
int ans = ksu( n, m );
if( ans == -1 ) System.out.println( "?" );
else System.out.println( ans );
}
sc.close();
}
public static void mst( int n ){
for( int i = 0; i <= n; i++ ){
pre[i] = i;
}
}
public static int find( int x ){
return x == pre[ x ] ? x : ( pre[ x ] = find( pre[ x ] ) );
}
public static int ksu( int n, int m ){
Arrays.sort( clg, 0, n, new mycmp() );
int cnt = 0;
int ans = 0;
for( int i = 0; i < n; i++ ){
int fu = find( clg[ i ].u );
int fv = find( clg[ i ].v );
if( fu != fv ){
ans += clg[i].w;
cnt ++;
pre [ fv ] = fu;
}
if( cnt == m - 1 ) return ans;
}
return -1;
}
}

HDU 1863 畅通工程 最小生成树的更多相关文章

  1. HDU 1863 畅通工程(最小生成树,prim)

    题意: 给出图的边和点数,要求最小生成树的代价,注:有些点之间是不可达的,也就是可能有多个连通图.比如4个点,2条边:1-2,3-4. 思路: 如果不能连通所有的点,就输出‘?’.之前以为每个点只要有 ...

  2. hdu 1863 畅通工程 (并查集+最小生成树)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1863 畅通工程 Time Limit: 1000/1000 MS (Java/Others)    M ...

  3. <hdu - 1863> 畅通工程 并查集和最小生成树问题

    本题链接:http://acm.hdu.edu.cn/showproblem.php?pid=1863  Problem Description: 省政府“畅通工程”的目标是使全省任何两个村庄间都可以 ...

  4. hdu 1863 - 畅通工程(MST)

    畅通工程 Time Limit: 1000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submis ...

  5. 题解报告:hdu 1863 畅通工程

    Problem Description 省政府“畅通工程”的目标是使全省任何两个村庄间都可以实现公路交通(但不一定有直接的公路相连,只要能间接通过公路可达即可).经过调查评估,得到的统计表中列出了有可 ...

  6. HDU 1863 畅通工程

    畅通工程 Time Limit: 1000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Total Submissi ...

  7. HDU 1863 畅通工程(Prim算法求解MST)

    题目: 省政府“畅通工程”的目标是使全省任何两个村庄间都可以实现公路交通(但不一定有直接的公路相连,只要能间接通过公路可达即可).经过调查评估,得到的统计表中列出了有可能建设公路的若干条道路的成本.现 ...

  8. HDU 1863 畅通工程 克鲁斯卡尔算法

    畅通工程 Time Limit: 1000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submis ...

  9. HDU 1863 畅通工程(Kruskal)

    畅通工程 Time Limit: 1000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Submi ...

随机推荐

  1. 使用java去对比2个带数学公式的字符串

    首先大家看到这个题目,可能会不屑一顾,呵呵,是的,起初我也认为这是个很简单的任务,当任务拿到手里后,经过我作为程序员来讲已经磨炼的无比通透的大脑来讲发现这其实是个坑. 故事的起因是这样的,想开发一款给 ...

  2. cloud-init简介及组件说明

    http://cloudinit.readthedocs.io/en/latest/topics/examples.html介绍:    cloud-init是专为云环境中虚拟机的初始化而开发的工具, ...

  3. Python 3.x的编码问题

    Python 3的源码.py文件的默认编码方式为UTF-8(Python 2.x的默认编码格式为unicode). encode的作用,使我们看到的直观的字符转换成计算机内的字节形式. decode刚 ...

  4. Block层也是有IO的优先级的

    ---恢复内容开始--- 今天查看iotop的原理,竟然发现了IO优先级一说,IO是block层cfs调度器中的概念 block层也有一个类似于CPU的调度算法 对进程分成三个级别:RT,BE,IDL ...

  5. 一些需要注意的ts

    写了一段时间ts,在从头学习一遍,温故而之新 ts的一些技巧 1.巧用注释 通过/** */形式的注释可以给 TS 类型做标记,编辑器会有更好的提示: /** A cool guy. */ inter ...

  6. InputStream 、 InputStreamReader 、 BufferedReader

    1.InputStream.OutputStream 处理字节流的抽象类 InputStream 是字节输入流的所有类的超类,一般我们使用它的子类,如FileInputStream等. OutputS ...

  7. 华中农业大学第四届程序设计大赛网络同步赛 J

    Problem J: Arithmetic Sequence Time Limit: 1 Sec  Memory Limit: 128 MBSubmit: 1766  Solved: 299[Subm ...

  8. VMware Storage VMotion概述及功能

    可以跨存储阵列实时迁移虚拟机磁盘文件.VMware Storage VMotion 使您可以在共享存储位置之间和跨共享存储位置重新分配虚拟机磁盘文件,同时保证连续的服务供应和事务处理的完整性. 1.可 ...

  9. Spring和ActiveMQ集成实现队列消息以及PUB/SUB模型

    前言:本文是基于Spring和ActiveMQ的一个示例文章,包括了Point-To-Point的异步队列消息和PUB/SUB(发布/订阅)模型,只是做了比较简单的实现,无任何业务方面的东西,作为一个 ...

  10. 兼容firstChild和firstElementChild

    1.nextSibling和nextElementSibling 顾名思义,就是找下一个节点 nextSibling在低版本浏览器中可以顺利找到下一个元素节点,而在正常浏览器中找到的包含中间空格. n ...