zoukankan      html  css  js  c++  java
  • 二分图最大权最小权完美匹配模板KM

    在网上找了一份挺好的模板,先标一下哦~链接君:http://blog.csdn.net/abcjennifer/article/details/5844579

    #include <iostream>
    #include <string.h>
    #include <algorithm>
    #include <iostream>
    using namespace std;
    int max(int a,int b)
    {return a<b?b:a;}
    
    int min(int a,int b)
    {return a<b?a:b;}
    
    const int size = 160;
    const int INF = 100000000;
    
    bool map[size][size];         // 二分图的相等子图, map[i][j] = true 代表Xi与Yj有边
    bool sx[size], sy[size]; // 标记在一次DFS中,Xi与Yi是否在交错树上
    int match[size];             // 保存匹配信息,其中i为Y中的顶点标号,match[i]为X中顶点标号
    
    bool DFS(int, const int);
    void KM_Perfect_Match(const int n, const int edge[][size])
    {
        int i, j;
        int lx[size], ly[size];   // KM算法中Xi与Yi的标号
        for(i = 0; i < n; i++)
        {
            lx[i] = -INF;
            ly[i] = 0;
            for(j = 0; j < n; j++)
            {
                lx[i] = max(lx[i], edge[i][j]);
            }
        }
        bool perfect = false;
        while(!perfect) 
        {
            // 初始化邻接矩阵
            for(i = 0; i < n; i++)
            {
                for(j = 0; j < n; j++) 
                {
                    if(lx[i]+ly[j] == edge[i][j])
                        map[i][j] = true;
                    else map[i][j] = false;
                }
            }
            // 匹配过程
            int live = 0;
            memset(match, -1, sizeof(match));
            for(i = 0; i < n; i++) {
                memset(sx, false, sizeof(sx));
                memset(sy, false, sizeof(sy));
                if(DFS(i, n)) live++;
                else {
                    sx[i] = true;
                    break;
                }
            }
            if(live == n) perfect = true;
            else {
                // 修改标号过程
                int ex = INF;
                for(i = 0; i < n; i++) 
                {
                    for(j = 0; sx[i] && j < n; j++) 
                    {
                        if(!sy[j]) 
                            ex = min(ex, lx[i]+ly[j]-edge[i][j]);
                    }
                }
                for(i = 0; i < n; i++) 
                {
                    if(sx[i]) lx[i] -= ex;
                    if(sy[i]) ly[i] += ex;
                }
            }
        }
    }
    
    bool DFS(int p, const int n)//find augment path from X[p]
    {
        int i;
        for(i = 0; i < n; i++)
        {
            if(!sy[i] && map[p][i]) 
            {
                sy[i] = true;
                int t = match[i];
                match[i] = p;
                if(t == -1 || DFS(t, n))
                {
                    return true;
                }
                match[i] = t;
                if(t != -1) sx[t] = true;
            }
        }
        return false;
    }
    
    int main()
    {
        int n, edge[size][size]; // edge[i][j]为连接Xi与Yj的边的权值
        int i;
        int m;
        /***************************************************
        *       record edge[i][j] as edge value between vertex i in X and vertex j in Y
        *       save n as vertexs need to be match (used in KM_Perfect_Match(n, edge);)
        ***************************************************/
        int s,d,pow;
    
        while(scanf("%d%d",&n,&m)!=EOF)
        {
            if(n==0 && m==0) break;
            memset(edge,0,sizeof(edge));
            /*最小权: 去掉memset(edge,0,sizeof(edge));改为如下:
            for(i=0;i<n;i++)
                for(int j=0;j<n;j++)
                    edge[i][j]=-INF;
            */
            memset(sx,0,sizeof(sx));
            memset(sy,0,sizeof(sy));
            memset(match,0,sizeof(0));
            while(m--)
            {
                scanf("%d%d%d",&s,&d,&pow);
                s--;d--;
                /*最小权:edge[s][d]=pow; 改为edge[s][d]= - pow;*/
                edge[s][d]=pow;
            }
            KM_Perfect_Match(n, edge);
            int cost = 0;
            for(i=0;i<n;i++) 
            {
                cost += edge[match[i]][i];
            }
            /*最小权:output 改为 -cost*/
            cout<<cost<<endl;
        }
        // cost 为最大匹配的总和, match[]中保存匹配信息
        return 0;
    }
    View Code

     感觉这个模板也不错,比较符合审美。。。

    #include <cstdio>
    #include <cstring>
    #include <cmath>
    #include <algorithm>
    
    using namespace std;
    const int maxn = 105;
    const int inf = 0x3f3f3f3f;
    
    int N, L[maxn], Lx[maxn], Ly[maxn], W[maxn][maxn], slack[maxn];
    bool S[maxn], T[maxn];
    
    int match (int u) {
        S[u] = true;
        for (int i = 1; i <= N; i++) if (!T[i]) {
            if (Lx[u] + Ly[i] == W[u][i]) {
                T[i] = true;
                if (!L[i] || match(L[i])) {
                    L[i] = u;
                    return true;
                }
            } else
                slack[i] = min(slack[i], Lx[u]+Ly[i]-W[u][i]);
        }
        return false;
    }
    
    void update () {
        int a = inf;
        for (int i = 1; i <= N; i++) if (!T[i])
            a = min(a, slack[i]);
    
        for (int i = 1; i <= N; i++) {
            if (S[i]) Lx[i] -= a;
            if (T[i]) Ly[i] += a;
        }
    }
    
    void KM () {
        for (int i = 1; i <= N; i++) {
            L[i] = Lx[i] = Ly[i] = 0;
            for (int j = 1; j <= N; j++)
                Lx[i] = max(Lx[i], W[i][j]);
        }
    
        for (int i = 1; i <= N; i++) {
            for (int j = 1; j <= N; j++) slack[j] = inf;
            while (true) {
                for (int j = 1; j <= N; j++) S[j] = T[j] = false;
                if (match(i)) break;
                else update();
            }
        }
    }
    
    void init () {
        for (int i = 1; i <= N; i++)
            for (int j = 1; j <= N; j++)
                W[i][j] = -inf;
        int u, v;
        for (int i = 1; i <= N; i++) {
            while (scanf("%d", &u) == 1 && u) {
                scanf("%d", &v);
                W[i][u] = max(W[i][u], -v);
            }
        }
    }
    
    int main () {
        while (scanf("%d", &N) == 1 && N) {
            init();
            KM();
            bool flag = false;
            for (int i = 1; i <= N; i++) {
                if (L[i] && W[L[i]][i] == -inf)
                    flag = true;
            }
    
            if (flag) printf("N
    ");
            else {
                int ans = 0;
                for (int i = 1; i <= N; i++)
                    ans += (Lx[i] + Ly[i]);
                printf("%d
    ", -ans);
            }
        }
        return 0;
    }
    View Code
  • 相关阅读:
    Daily Scrumming 2015.10.20(Day 1)
    Buaaclubs项目介绍
    [转载] Linux创建用户后,切换用户报This account is currently not available
    NetFPGA-1G-CML从零开始环境配置
    Digilent Xilinx USB Jtag cable
    OVS流表table之间的跳转
    Linux换源
    Scapy安装以及简单使用
    Do in SDN
    KMP算法
  • 原文地址:https://www.cnblogs.com/sunus/p/4836362.html
Copyright © 2011-2022 走看看