修路方案 118 (prim判断最小生成树的不唯一性)

142 篇文章 0 订阅
31 篇文章 0 订阅

修路方案

时间限制: 3000 ms  |           内存限制: 65535 KB
难度: 5
描述

南将军率领着许多部队,它们分别驻扎在N个不同的城市里,这些城市分别编号1~N,由于交通不太便利,南将军准备修路。

现在已经知道哪些城市之间可以修路,如果修路,花费是多少。

现在,军师小工已经找到了一种修路的方案,能够使各个城市都联通起来,而且花费最少。

但是,南将军说,这个修路方案所拼成的图案很不吉利,想让小工计算一下是否存在另外一种方案花费和刚才的方案一样,现在你来帮小工写一个程序算一下吧。

输入
第一行输入一个整数T(1<T<20),表示测试数据的组数
每组测试数据的第一行是两个整数V,E,(3<V<500,10<E<200000)分别表示城市的个数和城市之间路的条数。数据保证所有的城市都有路相连。
随后的E行,每行有三个数字A B L,表示A号城市与B号城市之间修路花费为L。
输出
对于每组测试数据输出Yes或No(如果存在两种以上的最小花费方案则输出Yes,如果最小花费的方案只有一种,则输出No)
样例输入
2
3 3
1 2 1
2 3 2
3 1 3
4 4
1 2 2
2 3 2
3 4 2
4 1 2
样例输出
No
Yes
//这个代码一直WA,但我一直找不出来
#include<stdio.h>
#include<string.h>
#include<algorithm>
#define MIN(a,b) a>b?b:a 
#define INF 0x3f3f3f3f
#define MAX(a,b) a>b?a:b
using namespace std;
int n,m;
int dis[600],vis[600],map[600][600];
void prim(int x)
{
	memset(vis,0,sizeof(vis));
	int i,j,k,min,sum=0,cnt;
	for(i=1;i<=n;i++)
	{
		dis[i]=map[1][i];
	}
	dis[x]=0;
	vis[x]=1;
	for(j=1;j<n;j++)
	{
		k=1;min=INF;cnt=0;
		for(i=1;i<=n;i++)
		{
			if(!vis[i]&&dis[i]<min)
			{
				min=dis[i];
				k=i;
			}
		}
		for(i=1;i<=n;i++)
		{
			if(vis[i]&&map[k][i]==min)
				cnt++;
		}
		if(cnt>2)
		{
			printf("Yes\n");
			return ;
		}	
		sum+=min;
		vis[k]=1;
		for(i=1;i<=n;i++)
		{
			if(!vis[i]&&dis[i]>=map[k][i])
				dis[i]=map[k][i];
		}
	}
	printf("No\n");
} 
int main()
{
	int t;
	scanf("%d",&t);
	while(t--)
	{
		int i,j;
		int a,b,c;		
		scanf("%d%d",&n,&m);
		memset(map,INF,sizeof(map));
		while(m--)
		{
			scanf("%d%d%d",&a,&b,&c);
			if(map[a][b]>c)
				map[a][b]=map[b][a]=c;
		}
		prim(1);
	}
	return 0;
}
//参考别人的代码,要用到次小生成树(还没看到过)
	
#include <iostream>
#include <stdio.h>
#include <string.h>
using namespace std;
#define maxN 510
#define MAX 0x0fffffff
#define MIN -0x0fffffff
int N,M,map[maxN][maxN],dis[maxN],maxlen[maxN][maxN],pre[maxN];
bool vis[maxN];
int prim()
{
    int i,j,k,minn,pr;
    memset(vis,false,sizeof(vis));
    for(i=1; i<=N; i++) 
	{
        dis[i]=map[1][i];
        pre[i]=1;
    }
    vis[1]=true;
    for(j=1; j<N; j++) 
	{
        minn = MAX;
        for(i=1; i <= N; i++)
        {
            if(!vis[i] && dis[i]<minn) 
			{
                minn=dis[k=i];
            }
        }
    	pr = pre[k];
       	maxlen[k][pr] = maxlen[pr][k] = map[k][pr];
	    for(i = 1; i <= N; i++)
    	    if(vis[i])
                maxlen[i][k]=maxlen[k][i]=max(maxlen[i][pr],maxlen[pr][k]);
       	vis[k]=true;
	    for(i=1; i<=N; i++)
	    {
            if(!vis[i]&&dis[i]>map[k][i]) 
			{
            	dis[i]=map[i][k];
               	pre[i]=k;
           	}
        }
    }
    for(i=1; i < N; i++)
    {
        for(j = i+1; j <= N; j++)
        {
            if(pre[i] == j|| pre[j] == i) 
				continue;
            else if(maxlen[i][j] == map[i][j]) 
				return 1;
		}
	}
    return 0;
}
int main() 
{
    int T;
    scanf("%d",&T);
    while(T--) 
	{
        int u,v,w;
        scanf("%d%d",&N,&M);
        for(int i = 0; i <= N; i++)
        {
            for(int j=0; j <= N; j++) 
			{
                map[i][j] = MAX;
                maxlen[i][j] = MIN;
            }
        }
        for(int i = 0; i < M; i++) 
		{
            scanf("%d%d%d",&u,&v,&w);
            map[u][v]=map[v][u]=w;
        }
        if(prim())
			printf("Yes\n");
        else 
			printf("No\n");
    }
    return 0;
}

 

//看到大神的对次小生成树的讲解,感觉很好,贴上代码,及思路。。。

算法核心思想:在prime求MST的过程中 用数组存储MST里面任意两点间的唯一的路中 权值最大的那条边的权值。最后枚举不在MST里面的边<i,,j>,判断<i,j>的权值 是否 和 MST里面 i 到 j 的最大权值相等,只要有一条边满足就可以说明MST不唯一。(因为我们可以用这条不在MST的边来 代替 在MST的边,这样MST肯定不唯一)



数组使用

prime 模版用到三个数组 low[] vis[] Map[][],不再详细解释。

MST[ i ][ j ] :  存储MST中i 到 j 的唯一的路中 权值最大的那条边的权值。

InMST[ i ][ j ] :  判断<i,j>这条边是否在MST中.

pre[ i ] :  记录 i 点的前驱  fa ,low[ i ] = Map[ fa ][ i ]


对于选入MST的点next

MST更新:MST[ next ][ j ] = max(MST[ pre[ next ] ][ j ], low[ next ])。( j 属于MST里面的点)


 

#include <cstdio>
#include <cstring>
#include <algorithm>
#define MAXN 500+10
#define INF 100000000
using namespace std;
int Map[MAXN][MAXN];
bool vis[MAXN];
int low[MAXN];//上面三个数组 求MST 专用
int MST[MAXN][MAXN];//MST中 i 和 j的最大边权
int pre[MAXN];//记录前驱
bool InMST[MAXN][MAXN];//标记该边是否在MST中
int N, M;
void init()
{
    for(int i = 1; i <= N; i++)
    {
        Map[i][i] = 0;
        for(int j = 1; j < i; j++)
            Map[i][j] = Map[j][i] = INF;
    }
}
void getMap()
{
    int a, b, c;
    for(int i = 1; i <= M; i++)
    {
        scanf("%d%d%d", &a, &b, &c);
        if(Map[a][b] > c)
            Map[a][b] = Map[b][a] = c;
    }
}
void solve()
{
    //prime求MST
    int mincost = 0;//最小代价
    int next, Min;
    memset(InMST, false, sizeof(InMST));
    memset(MST, 0, sizeof(MST));
    for(int i = 1; i <= N; i++)
    {
        low[i] = Map[1][i];
        vis[i] = false;
        pre[i] = 1;//每个点前驱
    }
    vis[1] = true;
    for(int i = 2; i <= N; i++)
    {
        Min = INF; next = -1;
        for(int j = 1; j <= N; j++)
        {
            if(!vis[j] && Min > low[j])
            {
                next = j;
                Min = low[j];
            }
        }
        mincost += Min;
        vis[next] = true;
        int fa = pre[next];//当前找到点的 前驱
        InMST[next][fa] = InMST[fa][next] = true;//在MST中
        for(int j = 1; j <= N; j++)
        {
            if(vis[j] && j != next)//MST中的点
                MST[j][next] = MST[next][j] = max(MST[fa][j], low[next]);
            if(!vis[j] && low[j] > Map[next][j])//更新low
            {
                low[j] = Map[next][j];
                pre[j] = next;//更新前驱
            }
        }
    }
    //判断MST是否唯一
    for(int i = 1; i <= N; i++)
    {
        for(int j = 1; j < i; j++)
        {
            if(Map[i][j] != INF && !InMST[i][j])//有边 且不在MST中
            {
                if(Map[i][j] == MST[i][j])
                {
                    printf("Yes\n");
                    return ;
                }
            }
        }
    }
    printf("No\n");
}
int main()
{
	int t;
	scanf("%d",&t);
	while(t--)
	{
		scanf("%d%d", &N, &M);
    	{
        	init();
	        getMap();
    	    solve();
    	}
	}   
    return 0;
}


 

 


                
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值