题意:

     判断最小树是否唯一。

思路:

     我用了两种方法,主要就是好久没敲了,找个水题练练手,第一种就是先一遍最小生成树,然后枚举最小生成树上的每一条边,然后取消这条边,在跑一遍最小生成树,就这样一直跑最小生成树,如果找到了一颗和之前的那个一样的,那么就是不唯一,第二种方法也是先最小树,然后枚举,在枚举的时候不是继续重新跑,而是断开当前边,把树分成两个集合<两次深搜实现>,然后在枚举这连个集合之间是否可以找到一个可以代替当前枚举的最小树的边,实现复杂度的话应该是第二种快点,但理论上也快不多少,只是为了练练手,在多说一句,第二种方法和求次小树的思路有点像,但是次小树可以再这个上面进行dp优化,其实这个题目也可以直接判断次小树是否等于最小树。好像有点说多了。

#include<stdio.h>

#include<string.h>

#include<algorithm>

#define N 110

using namespace std;

typedef struct

{

   int x ,y ,c;

}EDGE;

EDGE edge[N*N];

int  mer[N] ,mst[N];

int finds(int x)

{

   return x == mer[x] ? x : mer[x] = finds(mer[x]);

}

bool camp(EDGE a ,EDGE b)

{

   return a.c < b.c;

}

int MST(int n ,int m ,int co)

{

   for(int i = 1 ;i <= n ;i ++)mer[i] = i;

   int Ans = 0 ,sum = 0;

   for(int i = 1 ;i <= m ;i ++)

   {

      if(i == co) continue;

      int xx = finds(edge[i].x);

      int yy = finds(edge[i].y);

      if(xx != yy) 

      {

         Ans += edge[i].c ,sum ++ ;

         mer[xx] = yy;

         if(co == -1) mst[sum] = i;

      }

      if(sum == n - 1) break;

   }

   return Ans;



   

int main ()

{

   int t ,i ,n ,m;

   scanf("%d" ,&t);

   while(t--)

   {

      scanf("%d %d" ,&n ,&m);

      for(i = 1 ;i <= m ;i ++)

      scanf("%d %d %d" ,&edge[i].x ,&edge[i].y ,&edge[i].c);

      sort(edge + 1 ,edge + m + 1 ,camp);

      int Ans = MST(n ,m ,-1);

      for(i = 1 ;i < n ;i ++)

      {

         int tmp = MST(n ,m ,mst[i]);

         if(Ans == tmp) break;

      }

      i == n || m == n - 1? printf("%d\n" ,Ans) : puts("Not Unique!");

   }

   return 0;

}

      

     

      

      

      

   

#include<stdio.h>

#include<string.h>

#include<algorithm>

#define N 110

using namespace std;

typedef struct

{

   int x ,y ,c;

}EDGE;

typedef struct

{

   int to ,next;

}STAR;

EDGE edge[N*N];

STAR E[N*N];

int  map[N][N] ,mer[N] ,mark[N];

int list[N] ,tot ,mst[N];

int L[N] ,R[N] ,ll ,rr;

void add(int a, int b)

{

   E[++tot].to = b;

   E[tot].next = list[a];

   list[a] = tot;

}

int finds(int x)

{

   return x == mer[x] ? x : mer[x] = finds(mer[x]);

}

int minn(int x ,int y)

{

   return x < y ? x : y;

}

bool camp(EDGE a ,EDGE b)

{

   return a.c < b.c;

}

int MST(int n ,int m)

{

   for(int i = 1 ;i <= n ;i ++) mer[i] = i;

   int Ans = 0 ,sum = 0;

   memset(list ,0 ,sizeof(list)) ,tot = 1;

   for(int i = 1 ;i <= m ;i ++)

   {

      int xx = finds(edge[i].x);

      int yy = finds(edge[i].y);

      if(xx != yy) 

      {

         Ans += edge[i].c ,sum ++;

         mer[xx] = yy;

         mst[sum] = i;

         add(edge[i].x ,edge[i].y);

         add(edge[i].y ,edge[i].x);

      }

      if(sum == n - 1) break;

   }

   return Ans;

}

void DFS1(int x)

{

   for(int k = list[x] ;k ;k = E[k].next)

   {

      int to = E[k].to;

      if(mark[to]) continue;

      mark[to] = 1;

      L[++ll] = to;

      DFS1(to);

   }

}

void DFS2(int x)

{

   for(int k = list[x] ;k ;k = E[k].next)

   {

      int to = E[k].to;

      if(mark[to]) continue;

      mark[to] = 1;

      R[++rr] = to;

      DFS2(to);

   }

}

int main ()

{

   int t ,n ,m ,i ,j ,mk;

   scanf("%d" ,&t);

   while(t--)

   {

       scanf("%d %d" ,&n ,&m);

       for(i = 1 ;i <= n ;i ++)

       for(j = 1 ;j <= n ;j ++)

       map[i][j] = 100000000;

       for(i = 1 ;i <= m ;i ++)

       {

          scanf("%d %d %d" ,&edge[i].x ,&edge[i].y ,&edge[i].c); 

          int x = edge[i].x ,y = edge[i].y;

          map[x][y] = map[y][x] = minn(map[x][y] ,edge[i].c);

      }

      sort(edge + 1 ,edge + m + 1 ,camp);

      int Ans = MST(n ,m);

      if(m == n - 1)

      {

         printf("%d\n" ,Ans);

         continue;

      }

      mk = 0;

      for(i = 1 ;i < n && !mk;i ++)

      {

         memset(mark ,0 ,sizeof(mark));

         int l = edge[mst[i]].x ,r = edge[mst[i]].y;

         mark[l] = mark[r] = 1;

         ll = rr = 0;

         L[++ll] = l ,R[++rr] = r;    

         DFS1(l) ,DFS2(r);

         for(int j = 1 ;j <= ll && !mk;j ++)

         {

            for(int k = 1 ;k <= rr && !mk ;k ++)

            {

               if(L[j] == edge[mst[i]].x && R[k] == edge[mst[i]].y || L[j] == edge[mst[i]].y && R[k] == edge[mst[i]].x)

               continue;

               if(map[L[j]][R[k]] == edge[mst[i]].c) mk = 1;

            }

         }

      }

      mk ? printf("Not Unique!\n"): printf("%d\n" ,Ans);

   }

   return 0;

}

         

      

   

   

   

   

   

   

   

   

   

   

 

   

POJ1679判断最小生成树的唯一性的更多相关文章

  1. poj1679(判断最小生成树是否唯一)

    题意:给出n个点,m条边,要你判断最小生成树是否唯一. 思路:先做一次最小生成树操作,标记选择了的边,然后枚举已经被标记了的边,判断剩下的边组成的最小生成树是否与前面的相等,相等,则不唯一,否则唯一. ...

  2. POJ-1679 The Unique MST (判断最小生成树的唯一性)

    <题目链接> 题目大意: 给定一张无向图,判断其最小生成树是否唯一. 解题分析: 对图中每条边,扫描其它边,如果存在相同权值的边,则标记该边:用kruskal求出MST. 如果MST中无标 ...

  3. POJ1679 The Unique MST(Kruskal)(最小生成树的唯一性)

    The Unique MST Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 27141   Accepted: 9712 D ...

  4. POJ-1679 The Unique MST(次小生成树、判断最小生成树是否唯一)

    http://poj.org/problem?id=1679 Description Given a connected undirected graph, tell if its minimum s ...

  5. PTA 7-4 最小生成树的唯一性 (35分)

    PTA 7-4 最小生成树的唯一性 (35分) 给定一个带权无向图,如果是连通图,则至少存在一棵最小生成树,有时最小生成树并不唯一.本题就要求你计算最小生成树的总权重,并且判断其是否唯一. 输入格式: ...

  6. poj1679(最小生成树)

    传送门:The Unique MST 题意:判断最小生成树是否唯一. 分析:先求出原图的最小生成树,然后枚举删掉最小生成树的边,重做kruskal,看新的值和原值是否一样,一样的话最小生成树不唯一. ...

  7. POJ 1679 The Unique MST(判断最小生成树是否唯一)

    题目链接: http://poj.org/problem?id=1679 Description Given a connected undirected graph, tell if its min ...

  8. POJ 1679 The Unique MST (次小生成树 判断最小生成树是否唯一)

    题目链接 Description Given a connected undirected graph, tell if its minimum spanning tree is unique. De ...

  9. K - The Unique MST (最小生成树的唯一性)

    Given a connected undirected graph, tell if its minimum spanning tree is unique. Definition 1 (Spann ...

随机推荐

  1. 简单的webRTC连接测试

    webRTC WebRTC (Web Real-Time Communications) 是一项实时通讯技术,它允许网络应用或者站点,在不借助中间媒介的情况下,建立浏览器之间点对点(Peer-to-P ...

  2. PowerShell的使用

                一:基于winserver2008版本powershell2.0的升级(至4.0) (1)首先:查看各版本的Powershell版本,如下所示: (2)打开虚拟机winserv ...

  3. 快速了解C# 8.0中“可空引用类型(Nullable reference type)”语言特性

    Visual C# 8.0中引入了可空引用类型(Nullable reference type),通过编译器提供的强大功能,帮助开发人员尽可能地规避由空引用带来的代码问题.这里我大致介绍一下可空引用类 ...

  4. Mybatis中由于${}直接注入引发的问题

    一.问题引入 我们先来看这段代码,我想从取值为${category}的表中查询全部信息. @Mapper public interface CategoryMapper { @Select(" ...

  5. P1579_哥德巴赫猜想(JAVA语言)

    题目背景 1742年6月7日哥德巴赫写信给当时的大数学家欧拉,正式提出了以下的猜想:任何一个大于9的奇数都可以表示成3个质数之和.质数是指除了1和本身之外没有其他约数的数,如2和11都是质数,而6不是 ...

  6. Windows下解析命令行参数

    linux通常使用GNU C提供的函数getopt.getopt_long.getopt_long_only函数来解析命令行参数. 移植到Windows下 getopt.h #ifndef _GETO ...

  7. IDA报错fatal error before kernel init

    编写了一个IDA64插件,结果再打开IDA后报错fatal error before kernel init,然后闪退. 检查了一遍代码没发现有问题,后来发现是环境有一处配置错误, IDA64.exe ...

  8. vue+element+oss实现前端分片上传和断点续传

    纯前端实现: 切片上传 断点续传 .断点续传需要在切上上传的基础上实现 前端之前上传OSS,无需后端提供接口.先上完整代码,直接复制,将new OSS里的参数修改成自己公司OSS相关信息后可用,如遇问 ...

  9. DAOS 分布式异步对象存储|故障模型

    DAOS 依靠大规模分布式单端口存储.因此,每个 Target 实际上都是一个单独的失败点. DAOS 通过在不同的容错域中提供 Target 间的冗余来实现数据和元数据的可用性和持久性.DAOS 内 ...

  10. Java 并发编程 生产者消费者模式

    本文部分摘自<Java 并发编程的艺术> 模式概述 在线程的世界里,生产者就是生产数据的线程,消费者就是消费数据的数据.生产者和消费者彼此之间不直接通信,而是通过阻塞队列进行通信,所以生产 ...