原文链接http://www.cnblogs.com/zhouzhendong/p/8384699.html


题目传送门 - BZOJ1458


题意概括

  有一个M * N的棋盘,有的格子是障碍。现在你要选择一些格子来放置一些士兵,一个格子里最多可以放置一个士兵,障碍格里不能放置士兵。我们称这些士兵占领了整个棋盘当满足第i行至少放置了Li个士兵, 第j列至少放置了Cj个士兵。现在你的任务是要求使用最少个数的士兵来占领整个棋盘。


题解

  对于无解的判断很简单,首先暴力判一判。

  然后首先假装全部填满,那么总个数-最大的能够删除的兵就是答案。

  我们考虑怎样构图。

  考虑一个格子的删除会影响到他所在的行和列,于是我就想到了以下构图方案。

  点:源点、汇点、每行一个行节点,每列一个列节点,每个格子一个节点。

  边:对于所有的行节点x,从源点->行x连一条边,容量为当前行在至少保留Lx个士兵的前提下最多能删掉兵的个数。

    对于所有的列节点x,从列x->汇点连一条边,容量为当前列在至少保留Cx个士兵的前提下最多能删掉兵的个数。

    对于所有的格点,从它所在行向他连一条边,从他向他所在列连一条边,如果这个点是障碍,那么这两条边容量为0,否则为1 。

  然后答案就n*m-k-MaxFlow啦。


代码

#include <cstring>
#include <cstdio>
#include <cstdlib>
#include <cmath>
#include <algorithm>
using namespace std;
const int N=10205,M=2*2*N,INF=1e9;
struct edge{
int x,y,cap,flow,nxt;
};
struct gragh{
int cnt,fst[N],dist[N],n,S,T,num[N],cur[N],p[N];
int q[N],head,tail;
edge e[M];
void set(int _S,int _T,int _n){
S=_S,T=_T,n=_n,cnt=1;
memset(fst,0,sizeof fst);
}
void add(int a,int b,int c){
cnt++;
e[cnt].x=a,e[cnt].y=b,e[cnt].cap=c,e[cnt].flow=0;
e[cnt].nxt=fst[a],fst[a]=cnt;
cnt++;
e[cnt].x=b,e[cnt].y=a,e[cnt].cap=0,e[cnt].flow=0;
e[cnt].nxt=fst[b],fst[b]=cnt;
}
void bfs(){
memset(dist,-1,sizeof dist);
head=tail=dist[T]=0;
q[++tail]=T;
while (head<tail)
for (int x=q[++head],y,i=fst[x];i;i=e[i].nxt)
if ((i&1)&&dist[y=e[i].y]==-1)
dist[q[++tail]=y]=dist[x]+1;
for (int i=1;i<=n;i++)
if (dist[i]==-1)
dist[i]=n;
}
void init(){
bfs();
memset(num,0,sizeof num);
for (int i=1;i<=n;i++)
num[dist[i]]++,cur[i]=fst[i];
}
int Augment(int &x){
int ex_flow=INF;
for (int i=T;i!=S;i=e[p[i]].x)
if (e[p[i]].cap-e[p[i]].flow<=ex_flow)
ex_flow=e[p[i]].cap-e[p[i]].flow,x=e[p[i]].x;
for (int i=T;i!=S;i=e[p[i]].x)
e[p[i]].flow+=ex_flow,e[p[i]^1].flow-=ex_flow;
return ex_flow;
}
int ISAP(){
int x=S,y,MaxFlow=0;
init();
while (dist[S]<n){
if (x==T){
MaxFlow+=Augment(x);
continue;
}
bool found=0;
for (int i=cur[x];i;i=e[i].nxt)
if (dist[y=e[i].y]+1==dist[x]&&e[i].cap>e[i].flow){
cur[x]=p[y]=i,x=y,found=1;
break;
}
if (!found){
int d=n+1;
for (int i=fst[x];i;i=e[i].nxt)
if (e[i].cap>e[i].flow)
d=min(d,dist[e[i].y]+1);
if (!--num[dist[x]])
return MaxFlow;
num[dist[x]=d]++,cur[x]=fst[x],x=x==S?x:e[p[x]].x;
}
}
return MaxFlow;
}
}g;
const int NM=105;
int n,m,k,S,T,p[NM][NM],L[NM],C[NM],Lz[NM],Cz[NM];
int main(){
scanf("%d%d%d",&n,&m,&k);
memset(p,0,sizeof p);
for (int i=1;i<=n;i++)
scanf("%d",&L[i]);
for (int i=1;i<=m;i++)
scanf("%d",&C[i]);
for (int i=1,x,y;i<=k;i++){
scanf("%d%d",&x,&y);
p[x][y]=1;
}
memset(Lz,0,sizeof Lz);
memset(Cz,0,sizeof Cz);
for (int i=1;i<=n;i++)
for (int j=1;j<=m;j++){
p[i][j]^=1;
Lz[i]+=p[i][j],Cz[j]+=p[i][j];
}
for (int i=1;i<=n;i++)
if (Lz[i]<L[i]){
puts("JIONG!");
return 0;
}
for (int i=1;i<=m;i++)
if (Cz[i]<C[i]){
puts("JIONG!");
return 0;
}
g.set(S=n*m+n+m+1,T=n*m+n+m+2,n*m+n+m+2);
for (int i=1;i<=n;i++)
for (int j=1;j<=m;j++){
g.add(n*m+i,i*(m-1)+j,p[i][j]);
g.add(i*(m-1)+j,n*m+n+j,p[i][j]);
}
for (int i=1;i<=n;i++)
g.add(S,n*m+i,Lz[i]-L[i]);
for (int i=1;i<=m;i++)
g.add(n*m+n+i,T,Cz[i]-C[i]);
printf("%d",n*m-k-g.ISAP());
return 0;
}

  

BZOJ1458 士兵占领 网络流 最大流 SAP的更多相关文章

  1. bzoj1458: 士兵占领(最大流)

    题目描述 有一个M * N的棋盘,有的格子是障碍.现在你要选择一些格子来放置一些士兵,一个格子里最多可以放置一个士兵,障碍格里不能放置士兵.我们称这些士兵占领了整个棋盘当满足第i行至少放置了Li个士兵 ...

  2. bzoj1458: 士兵占领 网络流

    链接 https://www.lydsy.com/JudgeOnline/problem.php?id=1458 也可以去luogu 思路 想成倒着删去点,使得依旧满足覆盖!! 左边横,右边列,之间用 ...

  3. BZOJ1458:士兵占领(有上下界最小流)

    Description 有一个M * N的棋盘,有的格子是障碍.现在你要选择一些格子来放置一些士兵,一个格子里最多可以放置一个士兵,障碍格里不能放置士兵.我们称这些士兵占领了整个棋盘当满足第i行至少放 ...

  4. BZOJ 1458: 士兵占领( 网络流 )

    先判无解 把整个棋盘都放上士兵, 只需求最多可以拿走多少个士兵即可.每一行看做一个点r(i), 每一列看做一个点c(i) S->r(i), c(i)->T 连边, 容量为可以拿走的最大士兵 ...

  5. BZOJ1458 士兵占领 【带上下界网络流】

    题目链接 BZOJ1458 题解 对行列分别建边,拆点,设置流量下限 然后\(S\)向行连边\(inf\),列向\(T\)连边\(inf\),行列之间如果没有障碍,就连边\(1\) 然后跑最小可行流即 ...

  6. POJ1273 USACO 4.2.1 Drainage Ditches CodeVS1993草地排水 网络流 最大流 SAP

    欢迎访问~原文出处——博客园-zhouzhendong 去博客园看该题解 传送门 - POJ 传送门 - CodeVS 题意概括 给出一个图,告诉你边和容量,起点是1,汇点是n,让你求最大流. 题解 ...

  7. bzoj1458 士兵占领

    费用流,连下面几类边 1.s->s',流量为n*m,费用为0,表示最多可放置n*m个士兵 2.s'->行 (1)流量为a[i],费用为-n*m,表示必须在这一行放置a[i]个士兵. (2) ...

  8. HDU2732 Leapin' Lizards 网络流 最大流 SAP

    原文链接http://www.cnblogs.com/zhouzhendong/p/8362002.html 题目传送门 - HDU2732 题意概括 给你一个网格,网格上的一些位置上有一只蜥蜴,所有 ...

  9. bzoj 1458 士兵占领(最大流)

    [题意] n行m列,第i行必须放L[i],第j列必须放C[j],有障碍格,求满足条件至少需要放多少. [思路] 至少放多少等价于最多不放多少. 对行列分别建XY点,则连边(S,Xi,a)(Yi,T,b ...

随机推荐

  1. 切换目录查询目录 tcp

    服务器 import socket import os import json sk = socket.socket() sk.bind(('127.0.0.1',8080)) sk.listen() ...

  2. 用Python优雅的处理日志

    我们可以通过以下3种方式可以很优雅配置logging日志: 1)使用Python代码显式的创建loggers, handlers和formatters并分别调用它们的配置函数: 2)创建一个日志配置文 ...

  3. 洛谷P3676 小清新数据结构题 [动态点分治]

    传送门 思路 这思路好妙啊! 首先很多人都会想到推式子之后树链剖分+线段树,但这样不够优美,不喜欢. 脑洞大开想到这样一个式子: \[ \sum_{x} sum_x(All-sum_x) \] 其中\ ...

  4. aview安装和使用

    一.安装aalibwget https://sourceforge.net/projects/aa-project/files/latest/download?source=files --no-ch ...

  5. 终于,我还是下决心学Java后台了

    我没有什么本事,人也丑,也不会忽悠,只能硬着头皮学习了.最近计划学习Java后台,因为最近接了私活的问题,好多都要Java后台和前端一起做.平常我在做什么,当然是忙着赚钱了 除了敲代码,你还有什么副业 ...

  6. RCNN--目标检测

    原博文:http://www.cnblogs.com/soulmate1023/p/5530600.html 文章简要介绍RCNN的框架,主要包含: 原图-->候选区域生成-->对每个候选 ...

  7. Java 开源博客 Solo 2.5.0 发布

    Java 开源博客 Solo 2.5.0 发布 Solo 是一款一个命令就能搭建好的 Java 开源博客系统,如果你想开个独立博客,请一定不要错过! 2.5.0 版本主要支持了 Markdown/JS ...

  8. Confluence 6 配置 简易信息聚合(RSS)

    一个 Confluence 的管理员可以配置下面的 RSS 特性: Confluence 针对 RSS 聚合返回的最大项目数量. Confluence 针对 RSS 聚合允许的最大时间周期. 上面两个 ...

  9. linux学习笔记:第二单元 UNIX和Linux操作系统概述

    第二单元 UNIX和Linux操作系统概述 UNIX是什么 UNIX操作系统的特点 UNIX 与Linux的关系 GNU项目与自由软件 GUN计划 自由软件意味着什么 Linux简介 Linux是什么 ...

  10. ERROR 1045 (28000): Access denied for user 'mysql'@'localhost' (using password: YES

    一.有时可以直接输入命令: mysql进入数据库 启动数据库:# mysqld_safe & 二.查看用户命令: mysql> use mysql; Reading table info ...