Luogu P2754 星际转移问题

思路

首先,对于地球能否到达月球的问题,考虑使用并查集维护。

对于每艘飞船能够到达的站点,放进一个集合里,若两艘飞船的集合有交集,那么就合并两个集合,最后只要地球和月亮在同一个集合里即可到达。

然后就是多少天送完的问题。

想要最少天数送完,那么每次就尽可能地多送,考虑最大流。

那么怎么建图?

先放 \(n+4\) 个点( \(n\) 个空间站 + 地球和月球 + 源点 +汇点 ),想当然地,源点-->地球,容量为 \(k\) (初始时地球上的人数),月球指向汇点,容量 \(inf\) 。(这里为了方便,地球为 \(0\) ,月球为 \(n+1\) )

那么怎么表示太空船在各个太空站之间的移动呢?

分层。

按时间分层。举个例子:存在第 \(i\) 艘飞船的航线为 \((3,1,2)\) ,那么在第一个时刻,它在第 \(3\) 个空间站;在第二个时刻,它在第 \(1\) 个空间站;以此类推。于是我们连接第一时刻的 \(3\) 与第二时刻的 \(1\) ,即 \((1,3)\) --> \((2,1)\) ,容量为 \(h_i\) (第 \(i\) 艘飞船的容量)。当 \((1,3)\) 有流向 \((2,1)\) 的流时,就表示第一时刻到第二时刻,有人乘坐第 \(i\) 艘飞船从空间站 \(3\) 到了空间站 \(1\) 。

这样便得到了一个横向为时间,纵向为站点的分层图。

又因为地球、月球、空间站的容量无限,所以各个站点在每个时刻联通,即 \((1,x)\) --> \((2,x)\) --> \((3,x)\) --> \(\dots\) , \(x\) 为 \(n+2\) 个站点中的任意一个,相当于人停在该站点不动。双因为,无论哪个时刻到达月球都算送达,所以每个时刻的月球都向 \(t\) 连边,即 \((y,n+1)\) --> \(t\) ,\(y\) 为任意一个时刻。

最后从 \(1\) 开始枚举时间,直到源点流出的 \(k\) 份流量全部流入了汇点再跳出。

/*   卑劣的手打示意分层图
time-> ans
place da1 da2 ... ans
| earth 0_1 0_2 ... 0_ans
V station_1 1_1 1_1 ... 1_ans
. . . ... .
. . . ... .
n+1 moon n+1_1 n+1_2 ... n+1_ans
*/

样例建图如下:

直到第六个时刻,该网络的最大流才为 \(1\) ,共历时 \(5\) 个时刻,答案为 \(5\) 。

Code

#include<bits/stdc++.h>
//#define int long long //祖宗
#define mp(a,b) make_pair(a,b)
using namespace std;
inline int read()
{
int s=0,w=1;
char ch=getchar();
while(ch<'0'||ch>'9'){if(ch=='-') w=-1;ch=getchar();}
while(ch>='0'&&ch<='9') s=s*10+ch-'0',ch=getchar();
return s*w;
}
const int N=3000,M=5e5+5;
int first[N],nex[M],to[M],w[M],num=1;
int n,m,k,s,t,ans,Maxflow;
int h[N],r[N],p[25][25];
inline void add(int u,int v,int val)
{
nex[++num]=first[u];
first[u]=num;
to[num]=v;
w[num]=val;
}
inline void Add(int u,int v,int val)
{
add(u,v,val);
add(v,u,0);
}
namespace ISAP
{
int dep[N],gap[N],cur[N],NUM;
void bfs()
{
memset(dep,-1,sizeof(dep));
queue<int> q;
q.push(t);
dep[t]=0;gap[0]=1;
while(!q.empty())
{
int u=q.front();q.pop();
for(int i=first[u];i;i=nex[i])
{
int v=to[i];
if(dep[v]!=-1) continue;
dep[v]=dep[u]+1;
gap[dep[v]]++;
q.push(v);
}
}
}
inline int dfs(int u,int in)
{
if(u==t) return in;
int out=0;
for(int i=cur[u];i;i=nex[i])
{
cur[u]=i;
int v=to[i];
if(!w[i]||dep[v]!=dep[u]-1) continue;
int res=dfs(v,min(w[i],in-out));
w[i]-=res;
w[i^1]+=res;
out+=res;
if(out==in) return out;
}
gap[dep[u]]--;
if(!gap[dep[u]]) dep[s]=NUM+1;
dep[u]++;
gap[dep[u]]++;
return out;
}
void work(int t)
{
NUM=(n+1)*(t+1)+2;//当前网络的点的个数(供ISAP用,如果你是Dinic就无视它)
bfs();
while(dep[s]<NUM)
{
memcpy(cur,first,sizeof(first));
Maxflow+=dfs(s,1e9);
}
}
}
namespace UFS//并查集,判断是否联通
{
int fa[N];
void init(){for(int i=0;i<N;++i) fa[i]=i;}
inline int getfa(int u)
{
if(fa[u]==u) return u;
return fa[u]=getfa(fa[u]);
}
inline void merge(int u,int v)
{
int a=getfa(u),b=getfa(v);
if(a!=b) fa[a]=b;
}
}
inline int turn(int t,int p){return (t-1)*(n+2)+p;}//将t时p地的位置转化为点的编号
main()
{//0->地球
//n+1->月球
using namespace UFS;
init();
n=read(),m=read(),k=read();s=N-1,t=N-2;
for(int i=1;i<=m;++i)
{
h[i]=read();r[i]=read();
for(int j=1;j<=r[i];++j)
{
p[i][j]=read();//第i艘船第j个站点的位置为p[i][j]
if(p[i][j]==-1) p[i][j]=n+1;
if(j!=1)merge(p[i][j],p[i][j-1]);
//位置p[i][j]与位置p[i][j+1]之间可以通过第i艘船到达
}
}
if(getfa(0)!=getfa(n+1))
{
printf("0");
return 0;
}
int ans=2;//枚举时间,t==1时,人都在地球上
Add(s,turn(1,0),k);//从源点到t==1时的地球
Add(turn(1,n+1),t,1e9);
Maxflow=0;
while(1)
{
Add(turn(ans,n+1),t,1e8);//从当前时刻的月亮到汇点
for(int i=0;i<=n+1;++i) Add(turn(ans-1,i),turn(ans,i),1e9);//与上一时刻
for(int i=1;i<=m;++i)
{
int a=turn(ans,p[i][(ans-1)%r[i]+1]);//第i艘船第ans时刻所在的点
int b=turn(ans-1,p[i][(ans-2)%r[i]+1]);//第i艘船第ans-1时刻所在的点
//分析一下就理解了
Add(b,a,h[i]);
}
ISAP::work(ans);//不用重置,每次加完边后在残余网络上跑,将每次流量累积即为该网络的最大流
if(Maxflow>=k)
{
printf("%d",ans-1);
return 0;
}
ans++;
}
return 0;
}

Luogu P2754 星际转移问题的更多相关文章

  1. 洛谷 P2754 星际转移问题【最大流】

    判无解的方法非常粗暴:快T了还是没有合法方案,就是无解. 然后枚举答案,对于每一天都建一套太空站,s连地球,t连月球,上一天的太空站连向这一天的太空站,流量均为inf.然后对于每个飞船,上一天的停靠站 ...

  2. Luogu 2245 星际导航(最小生成树,最近公共祖先LCA,并查集)

    Luogu 2245 星际导航(最小生成树,最近公共祖先LCA,并查集) Description sideman做好了回到Gliese 星球的硬件准备,但是sideman的导航系统还没有完全设计好.为 ...

  3. 线性规划与网络流24题●09方格取数问题&13星际转移问题

    ●(做codevs1908时,发现测试数据也涵盖了1907,想要一并做了,但因为“技术”不佳,搞了一上午) ●09方格取数问题(codevs1907  方格取数3) 想了半天,也没成功建好图: 无奈下 ...

  4. [CTSC1999][网络流24题] 星际转移

    36. [CTSC1999][网络流24题] 星际转移 ★★★☆   输入文件:home.in   输出文件:home.out   简单对比时间限制:1 s   内存限制:128 MB «问题描述: ...

  5. 网络流24题之星际转移问题(洛谷P2754)

    洛谷 P2754 题目背景 none! 题目描述 由于人类对自然资源的消耗,人们意识到大约在 2300 年之后,地球就不能再居住了.于是在月球上建立了新的绿地,以便在需要时移民.令人意想不到的是,21 ...

  6. 【网络流24题】No. 13 星际转移问题 (网络判定 最大流)

    [题意] 由于人类对自然资源的消耗, 人们意识到大约在 2300 年之后, 地球就不能再居住了.于是在月球上建立了新的绿地,以便在需要时移民. 令人意想不到的是, 2177 年冬由于未知的原因, 地球 ...

  7. [CTSC1999]【网络流24题】星际转移

    Description 由于人类对自然资源的消耗,人们意识到大约在2300 年之后,地球就不能再居住了.于是在月球上建立了新的绿地,以便在需要时移民.令人意想不到的是,2177 年冬由于未知的原因,地 ...

  8. 【刷题】LOJ 6015 「网络流 24 题」星际转移

    题目描述 由于人类对自然资源的消耗,人们意识到大约在 2300 年之后,地球就不能再居住了.于是在月球上建立了新的绿地,以便在需要时移民.令人意想不到的是,2177 年冬由于未知的原因,地球环境发生了 ...

  9. 【luogu P2245 星际导航】 题解

    题目链接:https://www.luogu.org/problemnew/show/P2245 = 货车运输 被逼着写过mst+lca 后来成了mst+树剖 #include <cstdio& ...

随机推荐

  1. 聊聊java工程师换工作那些事

    最近有个读者在苦恼一件事,那就是有大公司在挖他,他要不要从所在的小公司,跳槽到大公司,前提是两家公司的待遇基本一致.由这个问题结合自己多年的工作经验,来谈谈java工程师要不要跳槽,何时跳槽,怎么跳槽 ...

  2. AI框架外部用户贡献代码

    AI框架外部用户贡献代码 概述 飞桨是百度自主研发的一款开源的深度学习框架,是主流深度学习框架中首个完全国产化的产品,已经在农业.医疗.林业.科研.服务等领域成功应用.无论是已入职场的深度学习从业者. ...

  3. Python基础_python的数据类型

    一.Python 的标准数据类型 二.以下单个实际举例分析数据类型 2.1 python的数字类型 Int:整型,Python3 整型是没有限制大小的,可以当作 Long 类型使用,所以 Python ...

  4. jemeter压测, 高级应用: 发1万个请求,每个请求参数都不同, 使用CSV数据文件配置

    今天接到一个压测任务, 数据源需要自己从测试环境库中取, 并且使用jemeter 请求, 每个请求参数都不相同 这里使用jemeter的 CSV数据文件来配置: 这样配置好后, 开始发送请求: csv ...

  5. 实验7、Django VS Flask VS Node:如何选择

    实验介绍 1. 实验内容 在本教程中,我们将详细介绍Django和Flask之间的比较.Flask和Django是基于Python的Web开发框架.许多正在朝着轻型微框架发展.这些框架敏捷,灵活,小巧 ...

  6. 会点自动化就要25k? 现在年轻人这么浮躁吗

    面试中一问元素定位就对答如流.一问实际项目框架如何搭建就避重就轻.含糊其辞,这样的自动化实战能力也能拿25K?静待下文: 一.为什么现在自动化测试工资那么高呢? 结合现在的职场环境与企业用人需求,自动 ...

  7. 【dp】动归总结

    原标题:[DP专辑]ACM动态规划总结 转载自 http://blog.csdn.net/cc_again?viewmode=list http://blog.csdn.net/cc_again/ar ...

  8. redhat6版本网卡绑定做bond

    1.编写bond0配置文件 cd /etc/sysconfig/network-scripts(进入网卡配置文件路径) vi ifc-bond0(编辑bond0的配置文件,具体如下) DEVICE=b ...

  9. StringUtils中的常量

    //空格字符串 public static final String SPACE = " "; //空字符串 public static final String EMPTY = ...

  10. 初学springboot

    现在总是与数据库和前端打交道,让我觉得好厌烦,还是喜欢敲代码.最近问了几个朋友,都说潮流要学springCloud,然后学springCloud又要先学springboot,所以这段时间我会慢慢把sp ...