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. Windows 10, version 21H1 ARM64

    请访问原文链接:https://sysin.org/article/windows-10-arm/,查看最新版.原创作品,转载请保留出处. Windows 10, version 21H1 (rele ...

  2. 利用js判断文件是否为utf-8编码

    常规方案 使用FileReader以utf-8格式读取文件,根据文件内容是否包含乱码字符�,来判断文件是否为utf-8. 如果存在�,即文件编码非utf-8,反之为utf-8. 代码如下: const ...

  3. 用OpenCV4实现图像的超分别率

    用OpenCV4实现图像的超分别率 本实验原文链接:· https://arxiv.org/pdf/1807.06779.pdf 原文摘要 单图像超分辨率(SISR)的主要挑战是如何恢复微小纹理等高频 ...

  4. deeplearning模型量化实战

    deeplearning模型量化实战 MegEngine 提供从训练到部署完整的量化支持,包括量化感知训练以及训练后量化,凭借"训练推理一体"的特性,MegEngine更能保证量化 ...

  5. 闵可夫斯基引擎Minkowski Engine

    闵可夫斯基引擎Minkowski Engine Minkowski引擎是一个用于稀疏张量的自动微分库.它支持所有标准神经网络层,例如对稀疏张量的卷积,池化,解池和广播操作.有关更多信息,请访问文档页面 ...

  6. python_selenium_键盘事件

    引言 ----在实际的web测试工作中,需要配合键盘按键来操作,webdriver的  keys()类提供键盘上所有按键的操作,还可以模拟组合键Ctrl+a,Ctrl+v等. 举例: #cording ...

  7. IOS小组件(8):App与Widget数据共享

    引言   Widget是一个迷你版的App,IOS有沙盒机制,不同App之间无法直接共享数据.组件和主App之间其实就是不同App的关系,所以也无法通过userdefaults.standard来传数 ...

  8. ES6中的新特性:Iterables和iterators

    目录 简介 什么是iteration Iterable对象 普通对象不是可遍历的 自定义iterables 关闭iterators 总结 简介 为了方便集合数据的遍历,在ES6中引入了一个iterat ...

  9. IDEA HTTP Client(史上最全)

    文章很长,建议收藏起来,慢慢读! 疯狂创客圈为小伙伴奉上以下珍贵的学习资源: 疯狂创客圈 经典图书 : <Netty Zookeeper Redis 高并发实战> 面试必备 + 大厂必备 ...

  10. 20201123 实验二《Python程序设计》实验报告

    20201123 2020-2021-2 <Python程序设计>实验报告课程:<Python程序设计>班级:2011姓名:晏鹏捷学号:20201123实验教师:王志强实验日期 ...