poj 3281

#include <cstdio>
#include <queue>
#include <cstring>
#include <cstdlib>
#include <iostream>
#include <algorithm>
using namespace std;
const int N = 88888*4;
const int oo = 0x3f3f3f3f;
int head[N], start, End, cnt, vis[N];
struct da
{
    int v, flow, next;
} as[N];
void add(int u, int v, int flow)
{
    as[cnt].v = v;
    as[cnt].flow = flow;
    as[cnt].next = head[u];
    head[u] = cnt++;
}
int bfs()
{
    queue<int >Q;
    memset(vis, 0, sizeof(vis));
    vis[start] = 1;
    Q.push(start);
    while(!Q.empty())
    {
       int u = Q.front();
       Q.pop();
       if(u == End) return 1;
       for(int j = head[u]; j != -1; j = as[j].next)
       {
           int v = as[j].v;
           if(!vis[v] && as[j].flow)
           {
               vis[v] = vis[u] + 1;
               Q.push(v);
           }
       }
    }
    return 0;
}
int dfs(int u, int Maxflow)
{
    int v, flow, uflow=0;
    if(u == End) return Maxflow;
    for(int j = head[u]; j != -1; j = as[j].next)
    {
        v = as[j].v;
        if(vis[v] == vis[u]+1 && as[j].flow)
        {
            flow = min(as[j].flow, Maxflow-uflow);
            flow = dfs(v, flow);
            as[j].flow -= flow;
            as[j^1].flow += flow;
            uflow += flow;
            if(uflow == Maxflow)break;
        }
    }
    if(uflow == 0)vis[u] = 0;
    return uflow;
}
int dinic()
{
    int ans = 0;
    while(bfs())
        ans += dfs(start, oo);
    return ans;
}
int main()
{
    int N, F, D, i, j, num, ans, fi, di;
    while(~scanf("%d %d %d", &N, &F, &D))
    {
        memset(head, -1, sizeof(head));
        cnt = 0;
        start = 0;
        End = F+D+2*N+2;
        for(i = 1; i <= N; i++)
        {
            scanf("%d %d", &fi, &di);
            for(j = 1; j <= fi; j++)
            {
                scanf("%d", &num);
                add(num, F+i, 1);
                add(F+i, num, 0);
            }
            for(j = 1; j <= di; j++)
            {
                scanf("%d", &num);
                add(F+i+N, F+2*N+num, 1);
                add(F+2*N+num, F+N+i, 0);
            }
            add(F+i, F+N+i, 1);
            add(F+N+i, F+i, 0);
        }
        for(i = 1; i <= F; i++)
        {
            add(start, i, 1);
            add(i, start, 0);
        }
        for(i = 1; i <= D; i++)
        {
            add(2*N+F+i, End, 1);
            add(End, 2*N+F+i, 0);
        }
        ans = dinic();
        printf("%d\n", ans);
    }
    return 0;
}
时间: 2024-10-14 10:55:22

poj 3281的相关文章

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 最大流+建图

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

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

POJ 3281:Dining(最大流)

http://poj.org/problem?id=3281 题意:有n头牛,f种食物,d种饮料,每头牛有fnum种喜欢的食物,dnum种喜欢的饮料,每种食物如果给一头牛吃了,那么另一个牛就不能吃这种食物了,饮料也同理,问最多有多少头牛可以吃到它喜欢的饮料和食物. 思路:一开始还以为二分匹配可以做,当然如果只有食物或者饮料其中一种就可以做.难点在于建图.看了下书,因为要保证经过牛的流量是1(每种食物对应分配给一头牛,每种饮料对应分配给一头牛,避免一头牛吃多份),所以要把牛拆成两个点.形成这样的路

POJ 3281 Dining (网络流最大流 拆点建图 Edmonds-Karp算法)

Dining Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 10159   Accepted: 4676 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 fabulo

POJ 3281 (最大流+匹配+拆点)

题目链接:http://poj.org/problem?id=3281 题目大意:有一些牛,一堆食物,一堆饮料.一头牛要吃一份食物喝一份饮料才算满足,而且牛对某些食物和饮料才有好感,问最多有多少头牛是满足的. 解题思路: 没有费用的匹配最大流题. 我一开始是这么考虑的,S->牛->食物->饮料->T,cap都是1,啊哈,多简单. 这样是不对的.因为牛比较挑剔,假设牛1喜欢食物1,而不喜欢饮料1,那么食物1和饮料1之间连不连边呢? 不连吧,其它牛怎么办? 连吧,牛1怎么办? 果断不能