1458: 士兵占据

Time Limit: 10 Sec  Memory Limit: 64 MB

Submit: 685  Solved: 398

[Submit][Status][

id=1458" style="color:blue; text-decoration:none">Discuss]

Description

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

Input

第一行两个数M, N, K分别表示棋盘的行数,列数以及障碍的个数。 第二行有M个数表示Li。 第三行有N个数表示Ci。

接下来有K行,每行两个数X, Y表示(X, Y)这个格子是障碍。

Output

输出一个数表示最少须要使用的士兵个数。

假设不管放置多少个士兵都没有办法占据整个棋盘,输出”JIONG!” (不含引號)

Sample Input

4 4 4

1 1 1 1

0 1 0 3

1 4

2 2

3 3

4 3

Sample Output

4

数据范围

M, N <= 100, 0 <= K <= M * N

HINT

Source

有源汇有上下界最小流问题

将行抽象为左部点,列抽象为右部点。

假设某个位置没有障碍。就从行相应的节点到列相应的节点连边。下界为0,上界为1。

在从源点向每行相应的节点连边,从每列相应的节点向汇点连边,下界均为该行或列的最少士兵数。上界均为正无穷。

最后求s到t的最小流。

#include<iostream>
#include<cstdio>
#include<cstring>
#include<cmath>
#include<cstdlib>
#include<algorithm>
#include<queue>
#define F(i,j,n) for(int i=j;i<=n;i++)
#define D(i,j,n) for(int i=j;i>=n;i--)
#define ll long long
#define pa pair<int,int>
#define maxn 300
#define maxm 100000
#define inf 1000000000
using namespace std;
int head[maxn],cur[maxn],dis[maxn],in[maxn];
int cnt=1,mx=0,maxflow,n,m,k,x,y,s,t,ss,tt;
bool a[105][105];
struct edge_type
{
int next,to,v;
}e[maxm];
inline int read()
{
int x=0,f=1;char ch=getchar();
while (ch<'0'||ch>'9'){if (ch=='-') f=-1;ch=getchar();}
while (ch>='0'&&ch<='9'){x=x*10+ch-'0';ch=getchar();}
return x*f;
}
inline void add_edge(int x,int y,int v)
{
e[++cnt]=(edge_type){head[x],y,v};head[x]=cnt;
e[++cnt]=(edge_type){head[y],x,0};head[y]=cnt;
}
inline void insert(int x,int y,int l,int r)
{
in[x]-=l;in[y]+=l;
if (r-l) add_edge(x,y,r-l);
}
inline void build()
{
F(i,1,tt)
{
if (in[i]>0) add_edge(ss,i,in[i]);
else if (in[i]<0) add_edge(i,tt,-in[i]);
}
}
inline bool bfs()
{
queue<int>q;
memset(dis,-1,sizeof(dis));
dis[s]=0;q.push(s);
while (!q.empty())
{
int tmp=q.front();q.pop();
if (tmp==t) return true;
for(int i=head[tmp];i;i=e[i].next) if (e[i].v&&dis[e[i].to]==-1)
{
dis[e[i].to]=dis[tmp]+1;
q.push(e[i].to);
}
}
return false;
}
inline int dfs(int x,int f)
{
if (x==t) return f;
int tmp,sum=0;
for(int &i=cur[x];i;i=e[i].next)
{
int y=e[i].to;
if (e[i].v&&dis[y]==dis[x]+1)
{
tmp=dfs(y,min(f-sum,e[i].v));
e[i].v-=tmp;e[i^1].v+=tmp;sum+=tmp;
if (sum==f) return sum;
}
}
if (!sum) dis[x]=-1;
return sum;
}
inline void dinic()
{
maxflow=0;
while (bfs())
{
F(i,1,tt) cur[i]=head[i];
maxflow+=dfs(s,inf);
}
}
inline void minflow()
{
s=ss;t=tt;
dinic();
int ans=e[cnt].v;
if (ans<mx)
{
printf("JIONG!\n");
return;
}
e[cnt].v=e[cnt^1].v=0;
s=n+m+2;t=n+m+1;
dinic();
printf("%d\n",ans-maxflow);
}
int main()
{
int tot=0;
m=read();n=read();k=read();
s=m+n+1;t=s+1;ss=t+1;tt=ss+1;
F(i,1,m){x=read();tot+=x;insert(s,i,x,inf);}
mx=max(mx,tot);
tot=0;
F(i,1,n){x=read();tot+=x;insert(i+m,t,x,inf);}
mx=max(mx,tot);
F(i,1,k)
{
x=read();y=read();
a[x][y]=true;
}
F(i,1,m) F(j,1,n) if (!a[i][j]) insert(i,j+m,0,1);
build();
add_edge(t,s,inf);
minflow();
}

最新文章

  1. css实现div的高度填满剩余空间
  2. 基本数据类型-列表_元组_字典_day4
  3. ACM程序对拍
  4. PHP Socket实现websocket(一)基本函数介绍
  5. android studio新项目时提示:Plugin is too old, please update to a more recent version
  6. memcache 缓存的批量删除方案(转)
  7. mongo 安装
  8. WPF - Build Error总结
  9. MyBatis 最常见错误,启动时控制台无限输出日志
  10. 慎重使用MySQL auto_increment
  11. Lua API 小记1
  12. Qt---Javascript/Qt交互、脚本化
  13. CBC翻转攻击(实验吧_简单的登陆题)
  14. python脚本对 mysql数据库进行增删改查操作
  15. 深入理解JAVA中的NIO
  16. Print all attributes and values in a Javascript Object
  17. 编译lua-5.3.5时出错解决方法
  18. iOS安全攻防(十八):数据保护API
  19. 慢速HTTP拒接服务攻击(DoS)复现
  20. eclipse导入/导出项目要注意三个地方

热门文章

  1. html body width height 100%使用
  2. xsy 1845 - GCD
  3. 【CF56E】Domino Principle(线性扫描,伪DP)
  4. noip 2010 关押罪犯 二分答案+二分图染色 || 并查集
  5. hdu4635 有向图最多添加多少边使图仍非强连通
  6. IOS-&lt;input&gt;表单元素只能读,设置readonly时光标仍然可见的解决办
  7. SQLite的sqlite_sequence表
  8. Linux环境下编译JDK
  9. SQL SERVER 工具
  10. JavaScript : 零基础打造自己的jquery类库