题意:

第一行输入m和n,m是猪圈的数量,n是顾客的数量,下面n行 第 i+1行表示第i个顾客 , 输入第一个数字表示有几把猪圈的钥匙,后面输入对应的猪圈,最后一个数字输入顾客想买几头猪.

建图:

设一个源点 s 和一个汇点 t,s 连接猪圈被第一个打开的顾客,权值为猪圈的猪数量,t 与顾客相连,权值为顾客想要的猪的数量,因为题上说迈克会根据下一个顾客的需求来将猪转移到合适的猪圈,所以顾客与同一猪圈排队的后面紧邻的顾客相连,权值为正无穷。

想一下感觉挺对的,s点有无数头猪,但是边的权值为猪圈的数量,说明顾客只能拿这么多,后面紧邻的顾客最然边权是无穷大,但被前面的顾客限制着,虽然顾客可以拿到猪圈里的猪,但是能带到 t 汇点的最多是自己想要的数量,所以求s到t的最大流。

注意:

书上的代码还要开一个数组记录残留网络什么的,我觉得太繁琐,直接在一个 e [ ] [ ]数组里面操作就好。注释的代码是自己建图错误了

关于网络流数组操作

为什么要正向边减去 流,反向边加上 流?

我觉得就像水流一样,从 s 流到 t 后,再沿着原路返回(进行加减操作),如果这个残余网络还有路径可以流,就根据这个残余网络找s到t的路径。

#include<stdio.h>
#include<vector>
#include<string.h>
#include<queue>
using namespace std;
int s,t,n,m,pig[1010],p[120],a[1010],v[1010][120],e[120][120],inf=0x3f3f3f3f;
int bfs()//广搜去找增广路,记录路径
{
int book[110];
for(int i=0; i<=t; i++)
book[i]=0,a[i]=-1;
queue<int>q;
q.push(s);
book[s]=1;
while(!q.empty())
{
int k=q.front();
q.pop();
if(k==t)
return 1;
for(int i=1; i<=t; i++)
{
if(e[k][i]&&!book[i])
{
a[i]=k;
book[i]=1;
q.push(i);
}
}
}
return 0;
}
int main()
{
while(~scanf("%d%d",&n,&m))
{
memset(v,0,sizeof(v));
memset(e,0,sizeof(e));
s=m+1,t=m+2;
int t1,t2;
for(int i=1; i<=n; i++)
scanf("%d",&pig[i]);//几号猪舍有几头猪
int last[1120],num,k;
memset(last,0,sizeof(last));//记录几号猪舍前面一个顾客是谁
for(int i=1;i<=m;i++)//开始建图
{
scanf("%d",&num);
for(int j=0;j<num;j++)
{
scanf("%d",&k);
if(last[k]==0)
e[s][i]+=pig[k];
else
e[last[k]][i]=inf;
last[k]=i;
}
scanf("%d",&e[i][t]);
}//建图完成
// for(int i=1;i<=t;i++)
// {
// for(int j=1;j<=t;j++)
// printf("%d ",e[i][j]);
// printf("\n");
// }
// for(int i=1; i<=m; i++)
// {
// scanf("%d",&t1);
// int k;
// for(k=1; k<=m; k++)
// if(v[i][k]==0)
// break;
// for(int j=1; j<=t1; j++)
// {
// scanf("%d",&t2);
// v[i][k++]=t2;
// }
// scanf("%d",&p[i]);//几号顾客买几只猪
// }
// // for(int i=1; i<=n; i++) //几号猪舍
// {
// for(int j=1; j<=m&&v[i][j]; j++) //第几位顾客
// {
// if(j==1)
// e[s][v[i][j]]+=pig[i];
// else
// e[v[i][j-1]][v[i][j]]=inf;
// }
// }
// for(int i=1; i<=m; i++)
// e[i][t]=p[i];//建图完成 int sum=0;
while(1)
{
int h=t,minn=inf;
if(!bfs())//直到找不到增广路为止
break;
while(a[h]!=-1)//找到所走路径上最短的边
{
if(e[a[h]][h]<minn)
minn=e[a[h]][h];
h=a[h];
}
sum+=minn;
h=t;
while(a[h]!=-1)//减去那个最短的边,再反向加上
{
e[a[h]][h]-=minn;
e[h][a[h]]+=minn;
h=a[h];
}
}
printf("%d\n",sum);
}
return 0;
}

PIGS POJ - 1149网络流(最短增广路---广搜) + 建图的更多相关文章

  1. 网络流 A - PIGS POJ - 1149 最大流

    A - PIGS POJ - 1149 这个题目我开始感觉很难,然后去看了一份题解,写的很好 https://wenku.baidu.com/view/0ad00abec77da26925c5b01c ...

  2. AC日记——pigs poj 1149

    POJ - 1149 思路: 最大流: 代码: #include <cstdio> #include <cstring> #include <iostream> # ...

  3. POJ 1149 网络流 合并建图

    这个题目我敲了一个简单的EK,这不是难点 难点在于建图,按题目的要求 每个猪圈和顾客都建点的话,那也太多了...我看了Edelweiss里面的缩点方法才建好的图,哎,惭愧啊 实际那些猪圈根本不需要单独 ...

  4. POJ 1161 Walls【floyd 以面为点建图】

    题目链接:http://poj.org/problem?id=1161 题目大意: 1.给出m个区域,n个俱乐部点.接下来是n个俱乐部点以及各个区域由什么点围成.求一个区域到各个俱乐部点的距离之和最小 ...

  5. PIGS POJ - 1149(水最大流)

    题意: 有M个猪圈,每个猪圈里初始时有若干头猪.一开始所有猪圈都是关闭的.依次来了N个顾客,每个顾客分别会打开指定的几个猪圈,从中买若干头猪.每个顾客分别都有他能够买的数量的上限.每个顾客走后,他打开 ...

  6. POJ 2226 Muddy Fields(二分匹配 巧妙的建图)

    Description Rain has pummeled the cows' field, a rectangular grid of R rows and C columns (1 <= R ...

  7. poj 1087 A Plug for UNIX(字符串编号建图)

    A Plug for UNIX Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 14862   Accepted: 5026 ...

  8. poj 1149 Pigs 网络流-最大流 建图的题目(明天更新)-已更新

    题目大意:是有M个猪圈,N个顾客,顾客要买猪,神奇的是顾客有一些猪圈的钥匙而主人MIRKO却没有钥匙,多么神奇?顾客可以在打开的猪圈购买任意数量的猪,只要猪圈里有足够数量的猪.而且当顾客打开猪圈后mi ...

  9. HDU3549 Flow Problem(网络流增广路算法)

    题目链接. 分析: 网络流增广路算法模板题.http://www.cnblogs.com/tanhehe/p/3234248.html AC代码: #include <iostream> ...

随机推荐

  1. Swift--struct与class的区别(汇编角度底层分析)

    概述 相对Objective-C, Swift使用结构体Struct的比例大大增加了,其中Int, Bool,以及String,Array等底层全部使用Struct来定义!在Swift中结构体不仅可以 ...

  2. Docker部署LAMP项目

    前言 之前我们学习了如何在Linux部署LAMP项目,今天我们来学习一下如何在Docker下部署LAMP项项目吧! Docker 要求 CentOS 系统的内核版本高于 3.10 ,查看本页面的前提条 ...

  3. 一份简明的 Base64 原理解析

    书接上回,在 记一个 Base64 有关的 Bug 一文里,我们说到了 Base64 的编解码器有不同实现,交叉使用它们可能引发的问题等等. 这一回,我们来对 Base64 这一常用编解码技术的原理一 ...

  4. Excel之在单元格中生成随机密码

    公式 =CHAR(INT(RAND()*26+97))&INT(RAND()*10)&CHAR(INT(RAND()*26+97))&INT(RAND()*10) 分析 CHA ...

  5. C#委托和事件的简单实例

    委托 C#里这个委托我的理解是可以看成是一个方法模板的类型.(不过并没有找到相关的理解 比如有几个返回值,参数列表类型相同的方法,就能用同个模板类型来表示,然后实例化一个委托类型就绑定上一个或多个方法 ...

  6. Java集合04——fail-fast&fail-safe 详解

    在前几个回合中,我们已经详细了解过了 Java 集合中的List.Set 和 Map,对这部分内容感兴趣的朋友可以关注我的公众号「Java面典」了解.今天我们将为各位介绍集合的失败机制--fail-f ...

  7. C语言 变量初始化二进制、八进制、十六进制

    int a1 = 10;   //十进制 int a2 = 0b10;  //二进制 int a3 = 010;    //八进制 int a4 = 0x10;  //十六进制 打印的结果:

  8. 个人项目(Word Count)

    一.Github项目地址 https://github.com/AllForward/GP_Homework/tree/master/个人项目 二.题目叙述 这个项目要求写一个命令行程序,模仿已有wc ...

  9. 「每天五分钟,玩转 JVM」:对象访问定位

    前言 在「对象内存布局」一节中,我们了解到对象头中包含了一个叫做类型指针的东西,即对象指向它的类元数据的指针,虚拟机通过这个指针来确定这个对象是哪个类的实例.但是,并不是所有的虚拟机都是这么去做的.不 ...

  10. mysql8 修改root密码

    Navicat工具里选中mysql数据库 执行: ALTER user 'root'@'localhost' IDENTIFIED BY 'newpassward'; //newpassward 新密 ...