zoukankan      html  css  js  c++  java
  • SGU 242 Student's Morning 网络流(水

    题目链接:点击打开链接

    题意:

    给定n个人,m个终点

    以下n行表示每一个人能够去m个点。

    每一个人仅仅能去一个点。

    输出随意一个方案使得每一个点至少有2个人到达。

    若存在输出m行,第一个数字表示i这个点来了几个人,后面是人的点标。


    思路:

    建一个二部图n-m,然后m到汇点限流2。推断是否满流,若不满流就无解。

    若满流则其它的人随便走。


    #include <stdio.h>
    #include <string.h>
    #include <iostream>
    #include <math.h>
    #include <queue>
    #include <set>
    #include <algorithm>
    using namespace std;
    typedef int ll;
    #define N 420
    #define M 100000
    #define inf 107374182
    struct Edge{
        ll from, to, cap, nex;
    }edge[M*2];
    ll head[N], edgenum;
    void add(ll u, ll v, ll cap, ll rw = 0){
        Edge E = {u, v, cap, head[u]};
        edge[edgenum] = E;
        head[u] = edgenum ++;
        Edge E2 = {v, u, rw, head[v]};
        edge[edgenum] = E2;
        head[v] = edgenum ++;
    }
    ll sign[N];
    bool BFS(ll from, ll to){
        memset(sign, -1, sizeof sign);
        sign[from] = 0;
        queue<ll>q;
        q.push(from);
        while( !q.empty() ) {
            ll u = q.front(); q.pop();
            for(ll i = head[u]; ~i; i = edge[i].nex)
            {
                ll v = edge[i].to;
                if(sign[v] == -1 && edge[i].cap){
                    sign[v] = sign[u] +1, q.push(v);
                    if(sign[to] != -1) return true;
                }
            }
        }
        return false;
    }
    ll Stack[N], top, cur[N];
    ll Dinic(ll from, ll to){
        ll ans = 0;
        while( BFS(from, to) )
        {
            memcpy(cur, head, sizeof head);
            ll u = from; top = 0;
            while(1)
            {
                if(u==to)
                {
                    ll flow = inf, loc;
                    for(ll i = 0; i < top; i++)
                        if(flow > edge[Stack[i]].cap)
                        {
                            flow = edge[Stack[i]].cap;
                            loc = i;
                        }
                    for(ll i = 0; i < top; i++)
                    {
                        edge[ Stack[i] ].cap -= flow;
                        edge[Stack[i]^1].cap += flow;
                    }
                    ans += flow;
                    top = loc;
                    u = edge[Stack[top]].from;
                }
                for(ll i = cur[u]; ~i; cur[u] = i = edge[i].nex)
                    if(edge[i].cap && (sign[u]+1 == sign[edge[i].to]))break;
                if(cur[u] != -1){
                    Stack[top++] = cur[u];
                    u = edge[cur[u]].to;
                }
                else
                {
                    if(top==0)break;
                    sign[u] = -1;
                    u = edge[Stack[--top]].from;
                }
            }
        }
        return ans;
    }
    void init(){memset(head, -1, sizeof head); edgenum = 0;}
    int n, m, from, to, to2, ans[N], G[N];
    vector<int>D[N];
    bool solve(){
        from = 0; to = n+m+1; to2 = to+1;
        init();
        for(int i = 1; i <= n; i++)
        {
            G[i] = -1;
            add(from, i, 1);
            int x, y; scanf("%d",&x);
            while(x--){
                scanf("%d",&y);
                G[i] = y;
                add(i, n+y, 1);
            }
        }
        for(int i = 1; i <= m; i++)
            add(n+i, to, 2);
        add(to, to2, inf);
        if(m*2 > n || Dinic(from, to2) < m*2)return false;
        puts("YES");
        for(int i = 1; i <= n; i++)
        {
            ans[i] = -1;
            for(int j = head[i]; ~j; j = edge[j].nex)
            {
                if(edge[j].cap==0 && edge[j].to != from)
                {
                    ans[i] = edge[j].to-n;
                    D[edge[j].to-n].push_back(i);
                    break;
                }
            }
            if(ans[i]==-1 && G[i]!=-1)
                D[G[i]].push_back(i);
        }
        for(int i = 1; i <= m; i++)
        {
            printf("%d", D[i].size());
            for(int j = 0; j < D[i].size(); j++)
                printf(" %d", D[i][j]);
            puts("");
        }
        return true;
    }
    int main(){
    	while(~scanf("%d %d",&n,&m)){
            if(solve()==false)puts("NO");
            for(int i = 1; i <= m; i++)D[i].clear();
    	}
    	return 0;
    }
    /*
    5 2
    1 1
    1 2
    1 1
    1 1
    1 1
    
    5 2
    1 1
    1 2
    1 1
    1 1
    2 1 2
    
    
    
    */


  • 相关阅读:
    【云栖社区001-数据结构】如何实现一个高效的单向链表逆序输出(Java版)
    全球疫情爬取及展示
    两个命令配置云服务器web开发环境
    云服务器的两类系统特点整理介绍
    团队介绍和项目简介
    Navicat连接阿里云服务器上MySQL数据库
    Android_bilibili式评论及回复的简单实现
    人月神话阅读笔记02
    每日进度博客_2
    每日进度博客_1
  • 原文地址:https://www.cnblogs.com/bhlsheji/p/5246507.html
Copyright © 2011-2022 走看看