zoukankan      html  css  js  c++  java
  • 求全局最小割(SW算法)

    hdu3002

    King of Destruction

    Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)
    Total Submission(s): 1022    Accepted Submission(s): 400


    Problem Description
    Zhou xingxing is the successor of one style of kung fu called "Karate Kid".he is falling love with a beautiful judo student,after being humiliated by her boyfriend,a Taekwando master from Japan,Zhou is going to fight with his rival in love.The way they fight is to destroy the wooden plank between some wooden pegs,in order to cut these wooden pegs into two disconnected parts,and destroy each piece of plank need consume different energy.However Zhou xingxing is beginner after all,so he is turn to you for help,please calculate the minimum energy he need to destroy the wooden plank.
     

    Input
    The input consists of multiple test cases.
    Each test case starts with two integers n (0 < n <= 100) and m in one line, where n、m are the number of wooden pegs and wooden plank. 
    Following are m lines, each line contains three integers s, e and q (0 <= s, e < n,q > 0), meaning that there need q energy to destroy the wooden plank between s and e.
     

    Output
    There is only one line for each test case, which contains the minimum energy they need to complete this fight.
     

    Sample Input
    2 1 0 1 50 3 2 0 1 50 1 2 10
     

    Sample Output
    50 10

    题意:求无向图的最小割。
    思路:无向图的最小割又叫全局最小割,枚举汇点求最大流效率很低,因而普遍使用StoerWagner算法,时间复杂度为O(n^3)。
    程序:
    #include"stdio.h"
    #include"string.h"
    #define inf 99999999
    int G[666][666],ans;
    int min(int a,int b)
    {
        return a<b?a:b;
    }
    int f[666];
    int finde(int x)
    {
        if(x!=f[x])
            f[x]=finde(f[x]);
        return f[x];
    }
    void make(int a,int b)
    {
        int x=finde(a);
        int y=finde(b);
        if(x!=y)
            f[x]=y;
    }
    void Mincut(int n)
    {
        ans=inf;
        int node[666],dis[666];
        bool use[666];
        int i,k,pre;
        for(i=0;i<n;i++)
            node[i]=i;
        while(n>1)
        {
            int maxj=1;
            for(i=1;i<n;i++)//初始化到已圈集合的割大小
            {
                dis[node[i]]=G[node[i]][node[0]];
                if(dis[node[maxj]]<dis[node[i]])
                {
                    maxj=i;
                }
            }
            pre=0;
            memset(use,false,sizeof(use));
            use[node[0]]=true;
            for(k=1;k<n;k++)
            {
                if(k==n-1)//只剩最后一个没加入集合的点,更新最小割
                {
                    ans=min(ans,dis[node[maxj]]);
                    for(i=0;i<n;i++)//合并最后一个点以及推出它的集合中的点
                    {
                        G[node[i]][node[pre]]=G[node[pre]][node[i]]+=G[node[i]][node[maxj]];
                    }
                    node[maxj]=node[--n];//缩点后的图
                }
                use[node[maxj]]=true;
                pre=maxj;
                maxj=-1;
                for(i=1;i<n;i++)
                {
                    if(!use[node[i]])
                    {
                        dis[node[i]]+=G[node[i]][node[pre]];//将上次求的maxj加入集合,合并与它相邻的边到割集
                        if(maxj==-1||dis[node[maxj]]<dis[node[i]])
                            maxj=i;
                    }
                }
            }
        }
    }
    int main()
    {
        int n,m,i;
        while(scanf("%d%d",&n,&m)!=-1)
        {
            memset(G,0,sizeof(G));
            for(i=0;i<n;i++)
            {
                f[i]=i;
            }
            while(m--)
            {
                int a,b,c;
                scanf("%d%d%d",&a,&b,&c);
                G[a][b]=G[b][a]+=c;
                make(a,b);
            }
            int flag=0;
            for(i=0;i<n;i++)
            {
                if(finde(i)!=finde(0))
                    flag++;
            }
            if(flag)
                printf("0
    ");
            else
            {
                Mincut(n);
                printf("%d
    ",ans);
            }
    
        }
    }
    



  • 相关阅读:
    使用eclipse阅读java源码
    找重复值
    mysql获取表中数据行数
    DBArtist之Oracle入门第2步: 了解Oracle的Database Control
    DBArtist之Oracle入门第1步: 如何安装Oracle 11g
    存储过程中使用事务和try catch
    Mysql设置auto_increment_increment和auto_increment_offset
    SQL SERVER性能优化综述
    Sql Server 2008R2 数据库发布与订阅
    无法修改SQL Server 服务名称
  • 原文地址:https://www.cnblogs.com/mypsq/p/4348236.html
Copyright © 2011-2022 走看看