bzoj1458: 士兵占领(最大流)
题目描述
有一个M * N的棋盘,有的格子是障碍。现在你要选择一些格子来放置一些士兵,一个格子里最多可以放置一个士兵,障碍格里不能放置士兵。我们称这些士兵占领了整个棋盘当满足第i行至少放置了Li个士兵, 第j列至少放置了Cj个士兵。现在你的任务是要求使用最少个数的士兵来占领整个棋盘。
输入输出格式
输入格式:
第一行两个数M, N, K分别表示棋盘的行数,列数以及士兵的个数。 第二行有M个数表示Li。 第三行有N个数表示Ci。 接下来有K行,每行两个数X, Y表示(X, Y)这个格子是障碍。
输出格式:
输出一个数表示最少需要使用的士兵个数。如果无论放置多少个士兵都没有办法占领整个棋盘,输出”JIONG!” (不含引号)
输入输出样例
说明
M, N <= 100, 0 <= K <= M * N Local
题解
据说正解是上下界网络流?还可以跑费用流?然而最大流也可以?
这里用的是最大流的做法
我们可以先在所有能摆的地方都摆上棋子,然后看一看最多能拿走多少棋子
给每行每列分别建一个点,如果$(x,y)$不是障碍格,就把$x$对应的点向$y$对应的点连边,容量为$1$表示这个点可以被删一次
然后从源点向所有行连边,容量为这一行最多能删的士兵数(总共格子数-障碍格子数-必须格子数)
从所有列向汇点连边,容量为这一列最多能删的士兵数(同上)
这样,可以发现不管怎么删都不会超出限制条件。那么要删掉最多士兵,只要跑一个最大流就可以了
- //minamoto
- #include<iostream>
- #include<cstdio>
- #include<cstring>
- #include<queue>
- #define inf 0x3f3f3f3f
- using namespace std;
- #define getc() (p1==p2&&(p2=(p1=buf)+fread(buf,1,1<<21,stdin),p1==p2)?EOF:*p1++)
- char buf[<<],*p1=buf,*p2=buf;
- inline int read(){
- #define num ch-'0'
- char ch;bool flag=;int res;
- while(!isdigit(ch=getc()))
- (ch=='-')&&(flag=true);
- for(res=num;isdigit(ch=getc());res=res*+num);
- (flag)&&(res=-res);
- #undef num
- return res;
- }
- const int N=,M=;
- int head[N],Next[M],ver[M],edge[M],tot=;
- int dep[N],cur[N],l[N],c[N],ll[N],cc[N],vis[N][N],n,m,k,s,t,ans;
- queue<int> q;
- inline void add(int u,int v,int e){
- ver[++tot]=v,Next[tot]=head[u],head[u]=tot,edge[tot]=e;
- ver[++tot]=u,Next[tot]=head[v],head[v]=tot,edge[tot]=;
- }
- bool bfs(){
- memset(dep,-,sizeof(dep));
- while(!q.empty()) q.pop();
- for(int i=s;i<=t;++i) cur[i]=head[i];
- q.push(s),dep[s]=;
- while(!q.empty()){
- int u=q.front();q.pop();
- for(int i=head[u];i;i=Next[i]){
- int v=ver[i];
- if(dep[v]<&&edge[i]){
- dep[v]=dep[u]+,q.push(v);
- if(v==t) return true;
- }
- }
- }
- return false;
- }
- int dfs(int u,int limit){
- if(u==t||!limit) return limit;
- int flow=,f;
- for(int i=head[u];i;i=Next[i]){
- int v=ver[i];
- if(dep[v]==dep[u]+&&(f=dfs(v,min(limit,edge[i])))){
- flow+=f,limit-=f;
- edge[i]-=f,edge[i^]+=f;
- if(!limit) break;
- }
- }
- if(!flow) dep[u]=-;
- return flow;
- }
- int dinic(){
- int flow=;
- while(bfs()) flow+=dfs(s,inf);
- return flow;
- }
- int main(){
- //freopen("testdata.in","r",stdin);
- n=read(),m=read(),k=read(),ans=n*m;
- s=,t=n+m+;
- for(int i=;i<=n;++i) l[i]=read();
- for(int i=;i<=m;++i) c[i]=read();
- for(int i=;i<=k;++i){
- int x=read(),y=read();vis[x][y]=;
- ++ll[x],++cc[y],--ans;
- }
- for(int i=;i<=n;++i)
- for(int j=;j<=m;++j)
- if(!vis[i][j]) add(i,j+n,);
- for(int i=;i<=n;++i){
- int p=m-ll[i]-l[i];
- if(p<) return puts("JIONG!"),;
- add(s,i,p);
- }
- for(int i=;i<=m;++i){
- int p=n-cc[i]-c[i];
- if(p<) return puts("JIONG!"),;
- add(i+n,t,p);
- }
- ans-=dinic();
- printf("%d\n",ans);
- return ;
- }
bzoj1458: 士兵占领(最大流)的更多相关文章
- 【BZOJ1458】士兵占领 最小流
[BZOJ1458]士兵占领 Description 有一个M * N的棋盘,有的格子是障碍.现在你要选择一些格子来放置一些士兵,一个格子里最多可以放置一个士兵,障碍格里不能放置士兵.我们称这些士兵占 ...
- 【BZOJ-1458】士兵占领 最大流
1458: 士兵占领 Time Limit: 10 Sec Memory Limit: 64 MBSubmit: 782 Solved: 456[Submit][Status][Discuss] ...
- BZOJ1458 士兵占领 网络流 最大流 SAP
原文链接http://www.cnblogs.com/zhouzhendong/p/8384699.html 题目传送门 - BZOJ1458 题意概括 有一个M * N的棋盘,有的格子是障碍.现在你 ...
- BZOJ1458:士兵占领(有上下界最小流)
Description 有一个M * N的棋盘,有的格子是障碍.现在你要选择一些格子来放置一些士兵,一个格子里最多可以放置一个士兵,障碍格里不能放置士兵.我们称这些士兵占领了整个棋盘当满足第i行至少放 ...
- 【bzoj1458】士兵占领(最大流||有源汇最大流)
转载 http://hzwer.com/2963.html Description 有一个M * N的棋盘,有的格子是障碍.现在你要选择一些格子来放置一些士兵,一个格子里最多可以放置一个士兵,障碍格里 ...
- bzoj 1458: 士兵占领 -- 最大流
1458: 士兵占领 Time Limit: 10 Sec Memory Limit: 64 MB Description 有一个M * N的棋盘,有的格子是障碍.现在你要选择一些格子来放置一些士兵 ...
- P4311 士兵占领[最大流]
题目地址 有一个$M * N$的棋盘,有的格子是障碍.现在你要选择一些格子来放置一些士兵,一个格子里最多可以放置一个士兵,障碍格里不能放置士兵.我们称这些士兵占领了整个棋盘当满足第i行至少放置了$L_ ...
- 【BZOJ1458】士兵占领 最大流的模板题
我们只要把他们可以有的限制用流量限制,再用两者关系限制一下就可以开心的跑了. #include <cstdio> #include <cstring> #include < ...
- bzoj1458 士兵占领
费用流,连下面几类边 1.s->s',流量为n*m,费用为0,表示最多可放置n*m个士兵 2.s'->行 (1)流量为a[i],费用为-n*m,表示必须在这一行放置a[i]个士兵. (2) ...
随机推荐
- [转帖]Linux systemd 常用命令
Linux systemd 常用命令 https://www.cnblogs.com/tsdxdx/p/7288490.html systemctl hostnamectl timedatectl l ...
- superset部署
superset功能概述: 丰富的数据可视化集 易于使用的界面,用于探索和可视化数据 创建和共享仪表板 与主要身份验证提供程序集成的企业级身份验证(通过Flask AppBuilder进行数据库,Op ...
- Vufuria入门 1 图片识别和选择
Vufutia中的图片识别功能,底层主要是识别特征点来实现的.特征点,即那些棱角分明的点.尖锐的而不是圆滑的.对比度大的而不是小的. *** 步骤: 进入vofuria官网,登录,点击develop. ...
- JavaScript之二分法
二分法: 二分查找,又称为折半查找. 注意:二分法查找的数组必须是有序的. /* 获取元素88在数组中第一次出现的索引位置 如果数组元素中存在88,则直接返回88在数组中的索引位置即可. 如果 ...
- 第5章:Linux系统管理
1.文件读写 1).Python内置的open函数 f = open('data.txt', 'w') f.write('hello, world') f.close() 2).避免文件句柄泄露 tr ...
- MySQL优化 - 性能分析与查询优化(转)
出处: MySQL优化 - 性能分析与查询优化 优化应贯穿整个产品开发周期中,比如编写复杂SQL时查看执行计划,安装MySQL服务器时尽量合理配置(见过太多完全使用默认配置安装的情况),根据应用负载 ...
- linux的定时器(timer_create,timer_gettime,timer_delete,SIGEV_SIGNAL)
ref : http://blog.chinaunix.net/uid-28458801-id-5035347.html 系统中的一个模块需要频繁的获取系统时间,使用linux中内置的函数开销过大 ...
- Spring Boot(一) 初步理解Spring Boot
一.Spring Boot所解决的问题 Java开发十分笨重:繁多的配置.低下的开发效率.复杂的部署流程以头疼的第三方技术集成. Spring Boot的理念:习惯优于配置——项目中存在大量的配置,此 ...
- log4j application.properties 配置文件
log4j.rootLogger = info,stdout log4j.appender.stdout = org.apache.log4j.ConsoleAppenderlog4j.appende ...
- 关于Mybatis的几件小事(二)
一.MyBatis缓存机制 1.简介 Mybatis包含了一个非常强大的查询缓存的特性,它可以非常方便地配置和定制. 缓存key极大提高查询效率 MyBatis系统中默认定义了两次缓存 默认情况下,只 ...