题意:

     判断最小树是否唯一。

思路:

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

                  0x00 实验环境 攻击机:Win 10.Win Server 2012 R2 靶机:Ubuntu18 (docker搭建的vulhub靶场) 0x01 影响版本 Apac ...

  2. 鹅厂二面,nginx回忆录

    前天二面鹅厂,面试官问出了"nginx你了解吗?"这样宽泛直白的句式,我一时抓不到重点,一时语噻. 下班想了一下,平时潜移默化用到不少nginx的能力,但在面试的时候没有吹成对应的 ...

  3. Redis不是一直号称单线程效率也很高吗,为什么又采用多线程了?

    Redis是目前广为人知的一个内存数据库,在各个场景中都有着非常丰富的应用,前段时间Redis推出了6.0的版本,在新版本中采用了多线程模型. 因为我们公司使用的内存数据库是自研的,按理说我对Redi ...

  4. Python深入:setuptools进阶

    作者:gqtcgq 来源:CSDN 原文:https://blog.csdn.net/gqtcgq/article/details/49519685 Setuptools是Python Distuti ...

  5. 如何在 ASP.Net Core 中实现 健康检查

    健康检查 常用于判断一个应用程序能否对 request 请求进行响应,ASP.Net Core 2.2 中引入了 健康检查 中间件用于报告应用程序的健康状态. ASP.Net Core 中的 健康检查 ...

  6. [换根DP]luogu P3647 [APIO2014]连珠线

    题面 https://www.luogu.com.cn/problem/P3647 不重复地取树中相邻的两条边,每次得分为两条边权和,问最大得分 分析 容易想到状态 f[i][0/1] 分别表示 i ...

  7. java例题_12 奖金问题(暴力破解)

    1 /*12 [程序 12 计算奖金] 2 题目:企业发放的奖金根据利润提成. 3 利润I低于或等于 10 万元时,奖金可提 10%: 4 利润高于 10 万元,低于 20 万元时,低于 10 万元的 ...

  8. 基于sklearn的波士顿房价预测_线性回归学习笔记

    > 以下内容是我在学习https://blog.csdn.net/mingxiaod/article/details/85938251 教程时遇到不懂的问题自己查询并理解的笔记,由于sklear ...

  9. BUUCTF 基础CODE REVIEW

    1.说明: 题目来自于BUUCTF 的基础部分,内容就如题,是一个代码审计.代码如下: <?php /** * Created by PhpStorm. * User: jinzhao * Da ...

  10. java面试-垃圾回收器谈谈你的理解

    一.垃圾回收算法: 引用计数 复制算法 标记-清除 标记-整理 二.垃圾回收的方式: 串行(Serial).并行(Parallel).并发(CMS).G1 1.串行垃圾回收器(Serial) 它为单线 ...