思路

如图,建立分层图跑dinic

每次在残余网络里加边继续跑

跑到ans>=k时候的i就是答案

诶呀啊,忘记弄箭头了,最后一列是向上的箭头,不过聪明的你们应该没啥影响

代码

#include <bits/stdc++.h>
#define FOR(i,a,b) for(int i=a;i<=b;++i)
using namespace std;
const int N = 1e6 + 7;
const int inf = 0x3f3f3f3f;
int read() {
int x=0,f=1;char s=getchar();
for(;s>'9'||s<'0';s=getchar()) if(s=='-') f = -1;
for(;s>='0'&&s<='9';s=getchar()) x = x*10 + s - '0';
return x*f;
}
int n, m, k, h[100], S, T;
std::vector<int> G[100];
struct node {
int v, nxt, cap;
}e[N];
int head[N], tot = 1;
void add_edge(int u, int v, int cap) {
e[++tot].v = v;
e[tot].cap = cap;
e[tot].nxt = head[u];
head[u] = tot;
}
int dis[N];
queue <int> Q;
bool bfs() {
memset(dis, -1, sizeof(dis));
while(!Q.empty()) Q.pop();
dis[S] = 0;
Q.push(S);
while(!Q.empty()) {
int u = Q.front();
Q.pop();
for(int i = head[u]; i;i = e[i].nxt) {
int v = e[i].v;
if(dis[v] == -1 && e[i].cap) {
dis[v] = dis[u] + 1;
Q.push(v);
}
}
}
return dis[T] != -1;
}
int dfs(int now, int f) {
if(now == T) return f;
int rest = f;
for(int i = head[now]; i;i = e[i].nxt) {
int v = e[i].v;
if(dis[v] == dis[now] + 1 && e[i].cap && rest) {
int t = dfs(v, min(e[i].cap, rest));
if(!t) dis[v] = 0;
e[i].cap -= t;
e[i^1].cap += t;
rest -= t;
}
}
return f-rest;
}
namespace bcj {
int siz[N], f[N];
int find(int x) {
return x == f[x] ? x : f[x] = find(f[x]);
}
void uu(int x, int y) {
int fx = find(x), fy = find(y);
if(siz[fx] <= siz[fy]) {
f[fx] = fy;
if(siz[fx] == siz[fy]) siz[fy] ++;
} else f[fy] = fx;
}
}
int main() {
n = read() ,m = read(), k = read();
S = 5e5+1, T = 5e5+2;
FOR(i,1,1000) bcj::f[i] = i;
FOR(i, 1, m) {
h[i] = read();
int x = read();
FOR(j, 1, x) {
int tmp = read();
if(tmp==0) tmp = n+1;
if(tmp==-1) tmp = n+2;
G[i].push_back(tmp);
}
}
FOR(i,1,m) {
for(std::vector<int>::iterator it = G[i].begin() + 1; it != G[i].end(); ++ it) {
bcj::uu(*it, *(it-1));
}
}
if(bcj::find(n+1) != bcj::find(n+2)) {puts("0"); return 0;}
int ans = 0,i = 1;
add_edge(S, n + 1, inf);
add_edge(n + 1, S, 0); add_edge(n + 2, T, inf);
add_edge(T, n + 2, 0);
for(;; ++i) {
// my -> my
FOR(j, 1, n + 1) {
add_edge(j + (i-1) * (n + 2), j + i * (n + 2), inf);
add_edge(j + i * (n + 2), j + (i-1) * (n + 2), 0);
}
add_edge((i + 1) * (n + 2), i * (n + 2), inf);
add_edge(i * (n + 2), (i + 1) * (n + 2), 0);
// go go go
FOR(j, 1, m) {
int a = i % G[j].size() - 1;
int b = i % G[j].size();
a= a<0 ? G[j].size()-1 : a;
a =G[j][a] + (i - 1) * (n + 2);
b =G[j][b] + i * (n + 2);
add_edge(a, b, h[j]);
add_edge(b, a, 0);
}
while(bfs()) ans += dfs(S,inf);
if(ans >= k) break;
}
printf("%d\n", i);
return 0;
}

网络流24题 P2754 [CTSC1999]家园的更多相关文章

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

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

  2. COGS743. [网络流24题] 最长k可重区间集

    743. [网络流24题] 最长k可重区间集 ★★★   输入文件:interv.in   输出文件:interv.out   简单对比时间限制:1 s   内存限制:128 MB «问题描述: «编 ...

  3. Cogs 14. [网络流24题] 搭配飞行员

    这道题其实蛮好想的,因为分为正,副飞行员.所以就把正飞行员当作Boy,副飞行员当作Girl.然后做Hungry即可. #include<bits/stdc++.h> using names ...

  4. 【线性规划与网络流 24题】已完成(3道题因为某些奇怪的原因被抛弃了QAQ)

    写在前面:SDOI2016 Round1滚粗后蒟蒻开始做网络流来自我拯救(2016-04-11再过几天就要考先修课,现在做网络流24题貌似没什么用←退役节奏) 做的题目将附上日期,见证我龟速刷题. 1 ...

  5. cogs_14_搭配飞行员_(二分图匹配+最大流,网络流24题#01)

    描述 http://cojs.tk/cogs/problem/problem.php?pid=14 有一些正飞行员和副飞行员,给出每个正飞行员可以和哪些副飞行员一起飞.一架飞机上必须一正一副,求最多多 ...

  6. BZOJ_1221_ [HNOI2001]_软件开发(最小费用流,网络流24题#10)

    描述 http://www.lydsy.com/JudgeOnline/problem.php?id=1221 n天,每天需要r个毛巾,用完以后可以洗,要么花fa洗a天,要么花fb洗b天,毛巾不够了可 ...

  7. 【网络流24题】 No.12 软件补丁问题(最小转移代价 最短路)

    [题意] T 公司发现其研制的一个软件中有 n 个错误, 随即为该软件发放了一批共 m 个补丁程序. 每一个补丁程序都有其特定的适用环境, 某个补丁只有在软件中包含某些错误而同时又不包含另一些错误时才 ...

  8. 【网络流24题】No.1 搭配飞行员(飞行员配对方案问题)

    [问题描述]     飞行大队有若干个来自各地的驾驶员,专门驾驶一种型号的飞机,这种飞机每架有两个驾驶员,需一个正驾驶员和一个副驾驶员.由于种种原因,例如相互配合的问题,有些驾驶员不能在同一架飞机上飞 ...

  9. CJOJ 1494 【网络流24题】 搭配飞行员(二分图最大匹配)

    CJOJ 1494 [网络流24题] 搭配飞行员(二分图最大匹配) Description 飞行大队有若干个来自各地的驾驶员,专门驾驶一种型号的飞机,这种飞机每架有两个驾驶员,需一个正驾驶员和一个副驾 ...

随机推荐

  1. “编程利器”:VSCode

    原先一直使用sublime text3,并且认为它是很好的编程利器. 但最近写代码时,发现很多代码还是提示的不够完整.我们知道,当代码名字很长时,还没有提醒,这是非常苦恼的一件事!同时它的调试功能也不 ...

  2. poj2987 求最大权闭合回路

    建图差不多和以前做的差不多,就是最后询问这个闭合子图有多少个的时候,只要输出这个图的S集合,就是进行dfs能遍历到的点一定在S集合中,不能遍历到的点在T集合中 #include <iostrea ...

  3. hdu5032 树状数组

    题意: 对于一个1000*1000的Mushroom, 起点在(1,1)给定一个斜率和一个x,求由斜率和x所对应的直线构成的三角形内蘑菇的总值. 每个点的对应的值为(x+A)(y+B) 解 每个点都有 ...

  4. 导入javax.servlet。伺服登记无法解决:The import javax.servlet.MultipartConfigElement cannot be resolved

    解决办法:

  5. Java多线程-----线程池详解

    1. 线程池的实现原理 提交一个任务到线程池中,线程池的处理流程如下: 判断线程池里的核心线程是否都在执行任务,如果不是(核心线程空闲或者还有核心线程没有被创建)则创建一个新的工作线程来执行任务.如果 ...

  6. Gambler Bo (高斯消元求特解)

    对于图中的每一个点假设点击Xi * m + j 然后每个点都有那么对于每一个点可以列举出一个方程式,n*m个点解n*m个未知数.利用高斯消元就可以解决. 问题就在这个题目可能不止有一个特,所以我们需要 ...

  7. (cvpr2019 ) Technology details of Deep Learning for Multiple-Image Super-Resolution

    Mutiple-Image SSR 关键的技术imformation fusion 1. 将单一场景的多图像经过Resnet, 其中每张图片的维度变为了输入的两倍.同时,这些输入的单一场景的多图像进行 ...

  8. 拜占庭将军问题(Byzantine Generals Problem),一个关于分布式系统容错问题故事

    拜占庭将军问题(Byzantine Generals Problem),一个关于分布式系统容错问题故事 背景:拜占庭帝国派出10支军队,去包围进攻一个强大的敌人,至少6支军队同时进攻才能攻下敌国. 难 ...

  9. HDU 1879 继续畅通工程(最小生成树)

    省政府“畅通工程”的目标是使全省任何两个村庄间都可以实现公路交通(但不一定有直接的公路相连,只要能间接通过公路可达即可).现得到城镇道路统计表,表中列出了任意两城镇间修建道路的费用,以及该道路是否已经 ...

  10. JustOj 1927: 回文串

    题目描述 回文串是从左到右或者从右到左读起来都一样的字符串,试编程判别一个字符串是否为回文串. 输入 输入一个字符串.串长度<255. 输出 判别输入的字符串是否为回文串,是输出"Y& ...