zoukankan      html  css  js  c++  java
  • ACM/ICPC 之 Dinic+枚举最小割点集(可做模板)(POJ1815)

        最小割的好题,可用作模板。

    //Dinic+枚举字典序最小的最小割点集
    //Time:1032Ms   Memory:1492K
    #include<iostream>
    #include<cstring>
    #include<cstdio>
    #include<algorithm>
    #include<queue>
    using namespace std;
    
    #define MAXN 205
    #define INF 0x3f3f3f3f
    
    int N, S, T;
    int s,t;
    int sres[2*MAXN][2*MAXN];   //source-res
    int res[2*MAXN][2*MAXN];
    int d[2*MAXN];
    int cut[MAXN];  //最小割点集
    bool v[2*MAXN];
    
    bool bfs()
    {
        memset(d, -1, sizeof(d));
        queue<int> q;
        q.push(s);  d[s] = 0;
        while(!q.empty() && d[t] == -1){
            int cur = q.front();    q.pop();
            for(int i = 1; i <= t; i++)
            {
                if(d[i] == -1 && res[cur][i])
                {
                    d[i] = d[cur] + 1;
                    q.push(i);
                }
            }
        }
        return d[t] != -1;
    }
    
    int dfs(int x, int sum)
    {
        if(x == t || sum == 0)  return sum;
        int src = sum;
        for(int i = 1; i <= t; i++)
        {
            if(d[i] == d[x] + 1 && res[x][i])
            {
                int tmp = dfs(i, min(sum, res[x][i]));
                res[x][i] -= tmp;
                res[i][x] += tmp;
                sum -= tmp;
            }
        }
        return src - sum;
    }
    
    int Dinic()
    {
        memcpy(res, sres, sizeof(sres));
        int maxFlow = 0;
        while(bfs())
            maxFlow += dfs(s,INF);
        return maxFlow;
    }
    
    int main()
    {
        //freopen("in.txt", "r", stdin);
    
        while(~scanf("%d%d%d", &N,&S,&T))
        {
            memset(sres, 0 ,sizeof(sres));
            s = 0;  t = 2*N+1;
            sres[0][S] = sres[T+N][t] = INF;
            for(int i = 1; i <= N; i++)
            {
                sres[i][i + N] = 1;
                for(int j = 1; j <= N; j++)
                {
                    int num;
                    scanf("%d", &num);
                    if(num && i != j) sres[i+N][j] = INF;
                }
            }
            sres[S][S+N] = sres[T][T+N] = INF;
            int ans = Dinic();
            if(ans == INF){ //不可分开
                printf("NO ANSWER!
    ");
                continue;
            }
            else {
                printf("%d
    ", ans);
                if(ans == 0) continue;  //已经分开
            }
    
            //枚举最小割点集
            int len = 0, tmp = ans;
            for(int i = 1; i <= N && tmp; i++)
            {
                if(i == S || i == T)   continue;
                if(res[i][i+N])    continue;
                sres[i][i+N] = 0;
                int k = Dinic();
                if(k != tmp){
                    tmp = k;
                    cut[len++] = i;
                }
                else sres[i][i+N] = 1;
            }
            for(int i = 0; i < ans - 1; i++)
                printf("%d ", cut[i]);
            printf("%d
    ", cut[ans-1]);
        }
    
        return 0;
    }
    
  • 相关阅读:
    SQL SERVER 2012修改数据库名称(包括 db.mdf 名称的修改)
    vmware三种网络模式
    指针
    linux 中 开放端口,以及防火墙的相关命令
    数据库备份的脚本,记录下,还需优化下
    遍历 目录的几种有效办法
    转。git 乌龟的使用安装
    centos 时区正确,时间不对
    locate
    从 零开始 无差错 装好nginx+PHP
  • 原文地址:https://www.cnblogs.com/Inkblots/p/5736556.html
Copyright © 2011-2022 走看看