题意:

     判断最小树是否唯一。

思路:

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

    前言 在面试中,Java 序列化被问到的几率还是挺高的.所以搜集了 Java 序列化常见的问题,由浅入深的帮助大家进一步学习和理解. 序列化基础知识 什么是序列化? Java 序列化是 JDK 1.1 ...

  2. 打造综合性智慧城市之朔州开发区 3D 可视化

      前言 近几年,我国智慧城市建设步伐也不断加快,党中央和国务院也更加注重智慧园区的建设与发展,智慧园区建设与园区产业发展相结合,向着创新化.生态化发展,更加注重高新技术.绿色环保型等产业的发展,将管 ...

  3. Less常用变量与方法记录

    需求:仅记录Lsee常用变量与方法定义,便于使用.-- @color: #000; @title-color: #000; @bg-color: #fff; @small-font: 12px; @l ...

  4. vue 倒计时 iOS无效

    vue实现的倒计时在苹果手机上无效,原因是因为后台返回的时间格式是'2019-1-29 17:13:04',而苹果手机只能解析这种时间格式'YYYY/MM/DD HH:mm:ss',修改后测试成功的代 ...

  5. 死磕生菜 -- lettuce 间歇性发生 RedisCommandTimeoutException 的深层原理及解决方案

    0x00 起源 项目的一些微服务集成了 Spring Data Redis,而底层的 Redis 客户端是 lettuce,这也是默认的客户端.微服务在某些环境中运行很正常,但在另一些环境中运行就会间 ...

  6. JavaCV 树莓派打造监控系统平台

    使用树莓派搭建视频监控平台去年就简单的实现了,只不过功能比较简陋,最近抽时间重构了原来的平台. 环境搭建 环境部分,参考旧版的安装及配置: 树莓派搭建视频监控平台 树莓派视频监控平台实现录制归档 框架 ...

  7. 自导自演的面试现场之--你竟然不了解MySQL的组提交?

    Hi,大家好!我是白日梦!本文是MySQL专题的第 26 篇. 下文还是白日梦以自导自演的方式,围绕"组提交"展开本话题.看看你能抗到第几问吧 换一种写作风格,自导自演面试现场!感 ...

  8. 【ZeyFraのJavaEE开发小知识05】Mybatis-Plus & Axios

    关于如何在Mybatis-Plus中添加SQL拦截器 之前ZeyFra在MyBatis-Plus[踩坑记录01]一文中提到过,使用Mybatis-Plus时最好使用MybatisSqlSessionF ...

  9. Bonuses on a Line Gym - 102569B

    题目链接:https://vjudge.net/problem/Gym-102569B 题意:数轴上有N个点,从0出发最多走t步问最多经过几个点. 思路:分开存负数点和整数点,然后枚举每个端点,某个点 ...

  10. Docker上安装Redis

    Docker可以很方便的进行服务部署和管理,下面我们通过docker来搭建Redis的单机模式.Redis主从复制.Redis哨兵模式.Redis-Cluster模式 一.在Docker上安装单机版R ...