POJ 3281 /// 最大流

题目大意:

n 头牛 f 种食物 d 种饮料

每头牛有各自喜欢的食物和饮料

求最多有多少头牛能分配到自己喜欢的食物和饮料

因为同时有食物和饮料 所以不能用二分图匹配

用最大流解决二分图匹配的办法

增加一个源点连向所有食物 每头牛与各自喜欢的食物连边

增加一个汇点连向所有的饮料 每头牛与各自喜欢的饮料连边

以上边容量都为1

单纯这样连的话 一头牛可能分配到多种食物和饮料

把一头牛拆成两个点 一点与食物连边 另一点与饮料连边

再在两个点之间连一条容量为1的边 这样就能保证只有一个流量流过

即只有一种食物被选 饮料反过来同理

此时从源点到汇点跑个最大流就能得到答案

#include <bits/stdc++.h>
#define INF 0x3f3f3f3f
using namespace std;
const int N=105;
bool F[N][N], D[N][N];
int n, f, d;

struct EDGE { int v,c,r; };
vector <EDGE> E[505];
bool vis[505];
void addE(int u,int v,int c) {
    E[u].push_back((EDGE){v,c,E[v].size()});
    E[v].push_back((EDGE){u,0,E[u].size()-1});
}
/**最大流*/
int dfs(int s,int t,int f) {
    if(s==t) return f;
    vis[s]=1;
    for(int i=0;i<E[s].size();i++) {
        EDGE& e=E[s][i]; // EDGE& 之后可直接修改到这个地址的数值
        if(!vis[e.v] && e.c>0) { // 没走过且可流过
            int d=dfs(e.v,t,min(f,e.c)); // 继续搜 能流过的最大流量
            if(d>0) { // 说明可流过d流量
                e.c-=d; // 那么这条边的剩余容量减小
                E[e.v][e.r].c+=d; // 反向边剩余容量变大
                return d;
            }
        }
    }
    return 0;
}
int maxFlow(int s,int t) {
    int flow=0;
    while(1) {
        memset(vis,0,sizeof(vis));
        int f=dfs(s,t,INF); // 边的容量改变后 继续搜
        // 边的容量改变后就可能会走反向边
        // 走反向边意味着 反悔正向的流量(可能有浪费)
        // 即 把正向的流量调小
        if(f==0) return flow; //  直到不可流
        flow+=f;
    }
}
/***/

/*  0~n-1 食物一侧的牛
    n~2*n-1 饮料一侧的牛
    2*n~2*n+f-1 食物
    2*n+f~2*n+f+d-1 饮料
*/
void solve() {
    // 增加一个源点 s=2*n+f+d
    // 增加一个汇点 t=s+1
    int s=2*n+f+d, t=s+1;
    for(int i=0;i<=t;i++) E[i].clear();
    for(int i=2*n;i<2*n+f;i++)
        addE(s,i,1); // 源点到食物
    for(int i=2*n+f;i<2*n+f+d;i++)
        addE(i,t,1); // 饮料到汇点
    for(int i=0;i<n;i++) {
        addE(i,i+n,1); // 一头牛拆成的两个点连边
        for(int j=0;j<f;j++)
            if(F[i][j]) addE(2*n+j,i,1); // 食物到牛的一点
        for(int j=0;j<d;j++)
            if(D[i][j]) addE(n+i,2*n+f+j,1); // 另一点到饮料
    }
    printf("%d\n",maxFlow(s,t));
}
int main()
{
    while(~scanf("%d%d%d",&n,&f,&d)) {
        memset(F,0,sizeof(F));
        memset(D,0,sizeof(D));
        for(int i=0;i<n;i++) {
            int a,b,t; scanf("%d%d",&a,&b);
            while(a--) {
                scanf("%d",&t); F[i][t-1]=1;
            }
            while(b--) {
                scanf("%d",&t); D[i][t-1]=1;
            }
        }
        solve();
    }

    return 0;
}

原文地址:https://www.cnblogs.com/zquzjx/p/10122294.html

时间: 2024-10-29 03:53:29

POJ 3281 /// 最大流的相关文章

poj 3281 最大流+建图

很巧妙的思想 转自:http://www.cnblogs.com/kuangbin/archive/2012/08/21/2649850.html 本题能够想到用最大流做,那真的是太绝了.建模的方法很妙! 题意就是有N头牛,F个食物,D个饮料. N头牛每头牛有一定的喜好,只喜欢几个食物和饮料. 每个食物和饮料只能给一头牛.一头牛只能得到一个食物和饮料. 而且一头牛必须同时获得一个食物和一个饮料才能满足.问至多有多少头牛可以获得满足. 最初相当的是二分匹配.但是明显不行,因为要分配两个东西,两个东

poj 3281 最大流拆点

Language: Default Dining Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 10321   Accepted: 4744 Description Cows are such finicky eaters. Each cow has a preference for certain foods and drinks, and she will consume no others. Farmer John

Dining POJ - 3281(最大流)

Dining Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 25452   Accepted: 11183 Description Cows are such finicky eaters. Each cow has a preference for certain foods and drinks, and she will consume no others. Farmer John has cooked fabul

poj 3281 Dining(最大流)

poj 3281 Dining Description Cows are such finicky eaters. Each cow has a preference for certain foods and drinks, and she will consume no others. Farmer John has cooked fabulous meals for his cows, but he forgot to check his menu against their prefer

POJ 3281 Dining(网络最大流)

http://poj.org/problem?id=3281 Dining Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 9121   Accepted: 4199 Description Cows are such finicky eaters. Each cow has a preference for certain foods and drinks, and she will consume no others.

POJ 3281 Dining(最大流建图 &amp;&amp; ISAP &amp;&amp; 拆点)

题目链接:http://poj.org/problem?id=3281 努力练建图ing!!! 题意:有 N 头牛,有 F 种食物和 D 种饮料,每种食物或饮料只能供一头牛享用,且每头牛只享用一种食物和一种饮料. 第2行-第N+1行.是牛i 喜欢A种食物,B种饮料,及食物种类列表和饮料种类列表. 问最多能使几头牛同时享用到自己喜欢的食物和饮料.->最大流. 本题难点是建图: 思路:一般都是左边一个集合表示源点与供应相连,右边一个集合表示需求与汇点相连. 但是本题,牛作为需求仍然是一个群体,但是供

POJ 3281 Dining(最大流)

POJ 3281 Dining 题目链接 题意:n个牛,每个牛有一些喜欢的食物和饮料,每种食物饮料只有一个,问最大能匹配上多少只牛每个牛都能吃上喜欢的食物和喜欢的饮料 思路:最大流,建模源点到每个食物连一条边,容量为1,每个饮料向汇点连一条边容量为1,然后由于每个牛有容量1,所以把牛进行拆点,然后食物连向牛的入点,牛的出点连向食物,跑一下最大流即可 代码: #include <cstdio> #include <cstring> #include <queue> #in

POJ 3281 网络流dinic算法

B - Dining Time Limit:2000MS     Memory Limit:65536KB     64bit IO Format:%I64d & %I64u Submit Status Practice POJ 3281 Appoint description: Description Cows are such finicky eaters. Each cow has a preference for certain foods and drinks, and she wil

POJ 3281 网络流 拆点保证本身只匹配一对食物和饮料

如何建图? 最开始的问题就是,怎么表示一只牛有了食物和饮料呢? 后来发现可以先将食物与牛匹配,牛再去和饮料匹配,实际上这就构成了三个层次. 起点到食物层边的容量是1,食物层到奶牛层容量是1,奶牛层到饮料层容量是1,饮料层到终点容量是1. 但是后来发现有一组hack数据: 2 3 3 3 3 1 2 3 1 2 3 3 3 1 2 3 1 2 3 我们发现一头奶牛居然吃了多个套餐,所以要解决这个只需要将自己与自己建立一条容量是1的边就行了. #include <cstdio> #include