• [ACM] FZU 2087 统计数边 (有多少边至少存在一个最小生成树里面)


    Problem Description

    在图论中,树:随意两个顶点间有且仅仅有一条路径的图。

    生成树:包括了图中全部顶点的一种树。

    最小生成树:对于连通的带权图(连通网)G,其生成树也是带权的。

    生成树T各边的权值总和称为该树的权,权最小的生成树称为G的最小生成树(Minimum Spanning Tree)。最小生成树可简记为MST。

    可是,对于一个图而言。最小生成树并非唯一的。

    如今,给你一个连通的有权无向图,图中不包括有自环和重边。你的任务就是寻找出有多少条边,它至少在一个最小生成树里。图保证连通。

     Input

    输入数据第一行包括一个整数T,表示測试数据的组数。对于每组測试数据:

    第一行包括两个整数n,m(1<n<100000,n-1<m<100000)。接下来m行,每行三个整数a,b,v(1<=a,b<=n,1<v<500),表示第i条路线连接景点A和景点B,距离是V。两个数字之间用空格隔开。

     Output

    对于每组測试数据,输出一行,包括一个整数,表示满足条件的边的个数。

     Sample Input

    14 51 2 1011 3 1002 3 22 4 23 4 1

     Sample Output

    4

     Source

    福州大学第九届程序设计竞赛

    解题思路:

    一棵树能够有多个最小生成树,可是他们的权值一定是相等的。

    按权值从小到大对边排序,然后把权值同样的边归为一组,推断时候,仅仅要一个边的两个端点不在同一个集合,那么该边就能够增加到最小生成树里面。

    代码:

    #include <iostream>
    #include <stdio.h>
    #include <string.h>
    #include <algorithm>
    using namespace std;
    const int maxn=100005;
    const int maxm=100005;
    int n,m;
    int parent[maxn];
    
    struct Node
    {
        int from,to,w;
    }node[maxm];
    
    void init(int n)
    {
        for(int i=1;i<=n;i++)
            parent[i]=i;
    }
    
    int find(int x)
    {
        if(parent[x]==x)
            return x;
        else
            return parent[x]=find(parent[x]);
    }
    
    bool same(int x,int y)
    {
        return find(x)==find(y);
    }
    void unite(int x,int y)
    {
        parent[find(x)]=find(y);
    }
    
    bool cmp(Node a,Node b)
    {
        return a.w<b.w;
    }
    
    void solve()
    {
        int cnt=0;
        int i,j;
        for(i=1;i<=m;i=j)
        {
            for(j=i;node[j].w==node[i].w;j++)//仅仅要同样权值的边两个端点不在同一个集合。该边就一定存在一个最小生成树里面
            {
                if(!same(node[j].from,node[j].to))
                    cnt++;
            }
            for(j=i;node[j].w==node[i].w;j++)//有回路的边加上不影响后面的加边
                unite(node[j].from,node[j].to);
        }
        printf("%d
    ",cnt);
    }
    
    int main()
    {
        int t;scanf("%d",&t);
        while(t--)
        {
            scanf("%d%d",&n,&m);
            init(n);
            for(int i=1;i<=m;i++)
                scanf("%d%d%d",&node[i].from,&node[i].to,&node[i].w);
            sort(node+1,node+1+m,cmp);
            solve();
        }
        return 0;
    }
    


  • 相关阅读:
    Winform中怎样设置ContextMenuStrip右键菜单的选项ToolStripMenuItem添加照片
    JavaScript垃圾回收机制
    前端如何处理内存泄漏
    前端缓存
    深入理解vue-router之keep-alive
    (淘宝无限适配)手机端rem布局详解
    mysql不会使用索引,导致全表扫描情况
    MYSQL性能优化的最佳20+条经验
    深拷贝与浅拷贝的区别,实现深拷贝的几种方法
    vue组件通信方式总结
  • 原文地址:https://www.cnblogs.com/jzssuanfa/p/7294968.html
Copyright © 2020-2023  润新知