CODEVS_1034 家园 网络流 最大流
原题链接:http://codevs.cn/problem/1034/
由于人类对自然的疯狂破坏,人们意识到在大约2300年之后,地球不能再居住了,于是在月球上建立了新的绿地,以便在需要时移民。令人意想不到的是,2177年冬由于未知的原因,地球环境发生了连锁崩溃,人类必须在最短的时间内迁往月球。
现有n个太空站处于地球与月球之间(编号1..n),m艘公共交通太空船在其中来回穿梭,每个太空站Si可容纳无限的人,每艘太空船pi只可容纳Hpi人。对于每一艘太空船pi,将周期性地停靠一系列的太空站(Si1,Si2…Sir),如:(1,3,4)表示停靠太空站1 3 4 1 3 4 1 3 4 …。 任一艘太空船从任一个太空站驶往另一个任意的太空站耗时为1。人只能在太空船停靠太空站(或地球、月球)时上船或下船。初始时的人全在地球上,太空船全在初始站(太空船pi处于Si1),目标是让所有的人尽快地全部转移到月球上。
文件第一行为三个正整数 n(太空站个数)、 m(太空船个数)、 k(需要运送的地球上的人的个数),其中 1<=m<=13, 1<=n<=20, 1<=k<=50。
接下来的n行给出了太空船的信息,第i+1行说明太空船pi,此行第一个数表示pi可容纳的人数Hpi,第二个数表示pi停靠一个周期的太空站个数r,1<=r<=n+2, 随后r个数便是停靠的太空站的编号(Si1,Si2,…,Sir), 地球用0表示,月球用-1表示。0时刻时,所有太空船都在初始站,随后开始运行,在时刻1,2,3…等正点时刻各艘太空船停靠相应的太空站,即人只有在0,1,2…等正点时刻才能上下太空船。
文件只有一个数,若问题有解,输出完成全部人员安全转移的时刻,否则输出0。
2 2 1
1 3 0 1 2
1 3 1 2 –1
5
1<=m<=13, 1<=n<=20, 1<=k<=50。
枚举时间t,每增加一单位时间,就增加一列点,这一列的点的序号是(0,t),(1,t)...(n,t),其中第一个数字表示太空站(0是地球,n是月球),第二个数字表示的是时间。然后从(i,t-1)到(i,t)连一条容量为INF的边(太空站可以停留无限的人),从地球(超级源点)连一条INF的边到(0,t),从(n,t)连一条INF的边到月球(超级汇点),然后考虑所有的太空船,设其中一个太空船的移动为a0,a1,a2....aT-1,那么连接一条从(a[(t-1)%T],t-1)到(a[t%T],t),容量为此太空船容量的边,求最大流,如果最大流的大于了总人数,就跳出,答案就是当前的时间。
详见代码:
#include<iostream>
#include<cstring>
#include<vector>
#include<queue>
#include<algorithm>
#include<string>
#include<cstdio>
#define MAX_M 201
#define MAX_V 100861
#define INF 10086111
using namespace std; struct edge{int to,cap,rev;};
vector<edge> G[MAX_V];
vector<int> ship[MAX_M];
//bool nodeColor[MAX_N];
const int moon=MAX_V-1,earth=MAX_V-2; int N,M,K; int shipCap[MAX_M];
int father[MAX_M]; bool used[MAX_V]; void init()
{
for(int i=0;i<=N+1;i++)father[i]=i;
} int Find(int u)
{
if(father[u]==u)return u;
else return father[u]=Find(father[u]);
} void unionSet(int u,int v)
{
int x=Find(u),y=Find(v);
if(x==y)return;
father[x]=y;
} bool Same(int u,int v)
{
return Find(u)==Find(v);
} bool check()
{
for(int i=0;i<M;i++)
{
int u=ship[i][0];
for(int j=1;j<ship[i].size();j++)
unionSet(u,ship[i][j]);
}
return Same(0,N+1);
} void add_edge(int from,int to,int cap)
{
G[from].push_back((edge){to,cap,G[to].size()});
G[to].push_back((edge){from,0,G[from].size()-1});
} int dfs(int v,int t,int f)
{
if(v==t)return f;
used[v]=1;
for(int i=0;i<G[v].size();i++)
{
edge &e=G[v][i];
if(!used[e.to]&&e.cap>0)
{
int d=dfs(e.to,t,min(f,e.cap));
if(d>0)
{
e.cap-=d;
G[e.to][e.rev].cap+=d;
return d;
}
}
}
return 0;
} int max_flow(int s,int t)
{
int flow=0;
while(1)
{
memset(used,0,sizeof(used));
int f=dfs(s,t,INF);
if(f==0)return flow;
flow+=f;
}
}
char cc;
int flo=0;
int main()
{
cin>>N>>M>>K;
for(int i=0;i<M;i++)
{
cin>>shipCap[i];
int r;
cin>>r;
for(int j=0;j<r;j++)
{
int ss;
cin>>ss;
if(ss==-1)ss=N+1;
ship[i].push_back(ss);
}
} if(!check()){cout<<0<<endl;return 0;} int nowNode=0;
N++;
int ans=0;
//cout<<N<<endl;
//cin>>cc>>cc;
for(int t=0;;t++)
{
for(int i=0;i<=N;i++)
{
int v=i+nowNode;
if(t!=0)
add_edge(v-N-1,v,INF);
if(i==0)
add_edge(earth,v,INF);
if(i==N)
add_edge(v,moon,INF);
}
for(int i=0;i<M&&t!=0;i++)
{
int tt=t%ship[i].size();
int fr=ship[i][(tt-1<0)?(tt-1+ship[i].size()):(tt-1)]+nowNode-N-1;
int go=ship[i][tt]+nowNode;
add_edge(fr,go,shipCap[i]);
}
flo+=max_flow(earth,moon); if(flo>=K){ans=t;break;}
nowNode+=N+1;
/*for(int i=0;i<nowNode;i++)
for(int j=0;j<G[i].size();j++)
cout<<i<<" "<<G[i][j].to<<" "<<G[i][j].cap<<endl;
cout<<flo<<endl;
cout<<"--------"<<endl;
cin>>cc;*/
if(t>100){cout<<0<<endl;return 0;}
}
cout<<ans<<endl;
return 0;
}
CODEVS_1034 家园 网络流 最大流的更多相关文章
- POJ 1459-Power Network(网络流-最大流-ISAP)C++
Power Network 时间限制: 1 Sec 内存限制: 128 MB 题目描述 A power network consists of nodes (power stations, cons ...
- [POJ1273][USACO4.2]Drainage Ditches (网络流最大流)
题意 网络流最大流模板 思路 EK也不会超时 所以说是一个数据比较水的模板题 但是POJ有点坑,多组数据,而且题目没给 哭得我AC率直掉 代码 用的朴素Dinic #include<cstdio ...
- HDU 3081 Marriage Match II (网络流,最大流,二分,并查集)
HDU 3081 Marriage Match II (网络流,最大流,二分,并查集) Description Presumably, you all have known the question ...
- HDU1532 网络流最大流【EK算法】(模板题)
<题目链接> 题目大意: 一个农夫他家的农田每次下雨都会被淹,所以这个农夫就修建了排水系统,还聪明的给每个排水管道设置了最大流量:首先输入两个数n,m ;n为排水管道的数量,m为节点的数量 ...
- Redraw Beautiful Drawings(hdu4888)网络流+最大流
Redraw Beautiful Drawings Time Limit: 3000/1500 MS (Java/Others) Memory Limit: 65536/65536 K (Java/O ...
- A simple Gaussian elimination problem.(hdu4975)网络流+最大流
A simple Gaussian elimination problem. Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/65 ...
- 【bzoj3130】[Sdoi2013]费用流 二分+网络流最大流
题目描述 Alice和Bob做游戏,给出一张有向图表示运输网络,Alice先给Bob一种最大流方案,然后Bob在所有边上分配总和等于P的非负费用.Alice希望总费用尽量小,而Bob希望总费用尽量大. ...
- 【bzoj1822】[JSOI2010]Frozen Nova 冷冻波 计算几何+二分+网络流最大流
题目描述 WJJ喜欢“魔兽争霸”这个游戏.在游戏中,巫妖是一种强大的英雄,它的技能Frozen Nova每次可以杀死一个小精灵.我们认为,巫妖和小精灵都可以看成是平面上的点. 当巫妖和小精灵之间的直线 ...
- 【bzoj1733】[Usaco2005 feb]Secret Milking Machine 神秘的挤奶机 二分+网络流最大流
题目描述 Farmer John is constructing a new milking machine and wishes to keep it secret as long as possi ...
随机推荐
- 【思维题】AGC013C - Ants on a Circle
妙妙技巧题 题目描述 题目大意 一个圆环上有n只蚂蚁,它们会按照顺时针或者逆时针行走.如果有蚂蚁相遇它们就会掉头(不一定在整数时间掉转).问最后每只蚂蚁的位置. 题目分析 以前在luogu上做过一道类 ...
- perl学习之子程序
一.定义子程序即执行一个特殊任务的一段分离的代码,它可以使减少重复代码且使程序易读.PERL中,子程序可以出现在程序的任何地方.定义方法为:sub subroutine{statements;}二.调 ...
- python--类的约束, 异常处理, MD5, 日志处理
一 . 类的约束 1. 写一个父类,父类中的某个方法要抛出一个异常 NotImplementedError class Base: # 对子类进行了约束. 必须重写该方法 # 以后上班了. 拿到公司代 ...
- Django 连接mysql数据库
首先在settings.py文件里将 DATABASES = { 'default': { 'ENGINE': 'django.db.backends.sqlite3', 'NAME': os.pat ...
- python基础——17(面向对象)
一.名称空间 名称空间有内置名称空间,全局名称空间,局部名称空间.它是用来存放名字与值对应关系的地方. test.py文件: num = 10 def fn(): print("fn run ...
- 【01】报错:webpack 不是内部或不可执行命令
[02] webpack 不是内部或不可执行命令 一般来安装完之后是可以直接执行的你可以执行 webpack -v 或者是 webpack --help 这样的就是正确的,我的问题的解决办法是 将 ...
- CEO的智力财富第12期-《股权激励》学习笔记
卷首语---你现在走的第一步,都藏着你未来的样子 今天,又去参加天使岛举办的系列讲座之股权激励,由律大大律师事务所李刚律师主讲,走在路上,我就在想,我为什么要来参加这样的活动呢?我的本职工作和股权没有 ...
- python基础-面向对象(类)
类 类的定义 >>> class P: ... pass ... >>> P <class __main__.P at 0x0000000001F4B ...
- 【LeetCode】Reorder Log Files(重新排列日志文件)
这道题是LeetCode里的第937道题. 题目描述: 你有一个日志数组 logs.每条日志都是以空格分隔的字串. 对于每条日志,其第一个字为字母数字标识符.然后,要么: 标识符后面的每个字将仅由小写 ...
- 【JavaScript 12—应用总结】:弹出登录框
导读:上篇博客中,做好了个人中心的下拉菜单,这次,将做每个网站都会有的一个登录功能,以此类推,可以做出别的想要的弹出框,如错误提示啦,或者注册. 一.实现分析 首先:和下拉菜单一样,需要通过CSS样式 ...