BZOJ 2756 奇怪的游戏(最大流)
题目链接:http://61.187.179.132/JudgeOnline/problem.php?id=2756
题意:在一个 N*M 的棋盘上玩,每个格子有一个数。每次 选择两个相邻的格子,并使这两个数都加上 1。 问最少多少次能使棋盘上的数都变成同一个数,如果永远不能变成同一个数则输出-1。
思路:对棋盘进行黑白染色,则每次操作使得黑白两色的格子总和各增加1。设黑色总和s1,个数cnt1;白色总和s2,个数cnt2,设最后的数字为x,那么有:
x*cnt1-s1=x*cnt2-s2。
(1)若cnt1!=cnt2,那么x是唯一的,x=(s1-s2)/(cnt1-cnt2),只要判定这个x是否合法即可;
(2)若cnt1=cnt2,那么x是不唯一的,但是x是单调的,即若x可以,则x+1也可以,因为黑白可以两两配对后各增加1。因此可以二分x,判定是否可行。
现在就是给定x判定是否可行。原点向黑格子连边,白格子向汇点连边,权值都是x-格子的数字。相邻格子连边无穷。满流即可。
struct node
{
int v,next;
i64 cap;
};
node edges[N];
int head[N],e;
void add(int u,int v,i64 cap)
{
edges[e].v=v;
edges[e].cap=cap;
edges[e].next=head[u];
head[u]=e++;
}
void Add(int u,int v,i64 cap)
{
add(u,v,cap);
add(v,u,0);
}
int pre[N],h[N],num[N],cur[N];
i64 Maxflow(int s,int t,int n)
{
int i;
for(i=0;i<=n;i++) h[i]=num[i]=0,cur[i]=head[i];
int u=s,v,x;
i64 ans=0,Min;
while(h[u]<n)
{
if(u==t)
{
Min=inf+1;
for(i=s;i!=t;i=edges[cur[i]].v)
{
x=cur[i];
if(edges[x].cap<Min) Min=edges[x].cap,v=i;
}
ans+=Min; u=v;
for(i=s;i!=t;i=edges[cur[i]].v)
{
x=cur[i];
edges[x].cap-=Min;
edges[x^1].cap+=Min;
}
}
for(i=cur[u];i!=-1;i=edges[i].next)
{
v=edges[i].v;
if(edges[i].cap>0&&h[u]==h[v]+1) break;
}
if(i!=-1)
{
cur[u]=i;
pre[edges[i].v]=u;
u=edges[i].v;
}
else
{
if(--num[h[u]]==0) break;
cur[u]=head[u];
x=n;
for(i=head[u];i!=-1;i=edges[i].next)
{
if(edges[i].cap>0) x=min(x,h[edges[i].v]);
}
h[u]=x+1;
num[x+1]++;
if(u!=s) u=pre[u];
}
}
return ans;
}
i64 a[55][55];
int n,m,b[55][55],s,t;
int dx[]={0,0,1,-1};
int dy[]={1,-1,0,0};
int check(i64 p)
{
clr(head,-1); e=0;
s=0; t=n*m+1;
int i,j,k,x,y;
i64 sum=0;
FOR1(i,n) FOR1(j,m)
{
if(p<a[i][j]) return 0;
if((i+j)&1) Add(s,b[i][j],p-a[i][j]),sum+=p-a[i][j];
else Add(b[i][j],t,p-a[i][j]);
if((i+j)%2==0) continue;
FOR0(k,4)
{
x=i+dx[k];
y=j+dy[k];
if(x>=1&&x<=n&&y>=1&&y<=m)
{
Add(b[i][j],b[x][y],inf);
}
}
}
return Maxflow(s,t,t+1)==sum;
}
i64 cal()
{
i64 s1=0,s2=0,cnt1=0,cnt2=0,Max=0,ans;
int i,j,k=0;
FOR1(i,n) FOR1(j,m) b[i][j]=++k;
FOR1(i,n) FOR1(j,m)
{
upMax(Max,a[i][j]);
if((i+j)%2==0) s1+=a[i][j],cnt1++;
else s2+=a[i][j],cnt2++;
}
if(cnt1!=cnt2)
{
ans=(s1-s2)/(cnt1-cnt2);
if(check(ans)) return ans*cnt1-s1;
return -1;
}
if(s1!=s2) return -1;
i64 low=Max,high=inf,mid;
ans=-1;
while(low<=high)
{
mid=(low+high)>>1;
if(check(mid)) ans=mid,high=mid-1;
else low=mid+1;
}
if(ans==-1) return -1;
return ans*cnt1-s1;
}
int main()
{
rush()
{
RD(n,m);
int i,j;
FOR1(i,n) FOR1(j,m) RD(a[i][j]);
PR(cal());
}
}
BZOJ 2756 奇怪的游戏(最大流)的更多相关文章
- bzoj 2756奇怪的游戏
2756: [SCOI2012]奇怪的游戏 Time Limit: 40 Sec Memory Limit: 128 MB Description Blinke 最近喜欢上一个奇怪的游戏. 这个游戏 ...
- [BZOJ 2756] 奇怪的游戏
Link:https://www.lydsy.com/JudgeOnline/problem.php?id=2756 Algorithm: 比较新颖的题目 首先发现是对矩阵中相邻两数进行操作 & ...
- BZOJ 2756 SCOI2012 奇怪的游戏 最大流
题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=2756 Description Blinker最近喜欢上一个奇怪的游戏. 这个游戏在一个 N ...
- BZOJ 2756: [SCOI2012]奇怪的游戏 [最大流 二分]
2756: [SCOI2012]奇怪的游戏 Time Limit: 40 Sec Memory Limit: 128 MBSubmit: 3352 Solved: 919[Submit][Stat ...
- 【BZOJ-2756】奇怪的游戏 最大流 + 分类讨论 + 二分
2756: [SCOI2012]奇怪的游戏 Time Limit: 40 Sec Memory Limit: 128 MBSubmit: 2925 Solved: 792[Submit][Stat ...
- BZOJ2756:[SCOI2012]奇怪的游戏(最大流,二分)
Description Blinker最近喜欢上一个奇怪的游戏. 这个游戏在一个 N*M 的棋盘上玩,每个格子有一个数.每次 Blinker 会选择两个相邻 的格子,并使这两个数都加上 1. 现在 B ...
- BZOJ2756 [SCOI2012]奇怪的游戏 最大流
好久没有写博客了.不过这个博客也没有多少人看 最近在写网络流,为了加深理解,来写一两篇题解. 对整个棋盘进行黑白染色以后可以发现,一次操作就是让二分图的两个点的值分别 \(+1\). 这样,我们就可以 ...
- 【BZOJ】【2756】【SCOI2012】奇怪的游戏
网络流-最大流 这题……建模部分先略过 这道题是会卡时限的T_T俺的Dinic被卡了,在此放几篇很棒的讲网络流算法的文章,至于大家耳熟能详的论文就不放了…… http://www.cppblog.co ...
- BZOJ 2756: [SCOI2012]奇怪的游戏 网络流/二分
2756: [SCOI2012]奇怪的游戏 Time Limit: 40 Sec Memory Limit: 128 MBSubmit: 1594 Solved: 396[Submit][Stat ...
随机推荐
- 【crunch bang】 tint2-用来控制桌面的布局
tint2配置: #--------------------------------------------- # TINT2 CONFIG FILE #----------------------- ...
- 【海岛帝国系列赛】No.5 海岛帝国:独立之战
50229234海岛帝国:独立之战 [试题描述] 恐怖分子多年来一直如饥似渴地渴求“药师傅”帝国,但是,“里脊肉”BANNIE时刻在守护着这一方水土.从而使帝国日益强大.如今,BANNIE由于在 “牡 ...
- android 项目学习随笔十八(三级缓存)
xUtils的BitmapUtils模块用的就是三级缓存,在项目中尽量还是应用BitmapUtils 三级缓存(机制) import com.itheima.zhsh66.R; import andr ...
- java中==与equal()方法的区别
==比较的是对象的地址,也就是是否是同一个对象: equal比较的是对象的值. Integer r1 = new Integer(900);//定义r1整型对象Integer r2 = new Int ...
- LoadRunner并发用户和集合点的深入讨论
看到51上三个高手Zee, 大漠飞鹰,xingcyx的一场非常精彩的关于并发用户数和集合点的讨论,很有意义.如果对这两个概念不清楚的朋友,一定要仔细领悟了. 故事开始于xingcyx的一番话: 声明: ...
- Hibernate,Session方法使得java对象进入持久化状态;持久化对象特征
以下情况java对象进入持久化状态: session.save()方法把临时对象转变为持久化对象. session.load()和session.get()方法得到的对象总是处于持久化状态. sess ...
- Linux之Ganglia源码安装
一.Ganglia简介: Ganglia是UC Berkeley发起的一个开源集群监视项目,设计用于测量数以千计的节点.Ganglia的核心包含gmond.gmetad以及一个Web前端.主要是用来监 ...
- 【PHP设计模式 03_JianDanGongChang.php】 简单工厂
<?php /** * [简单工厂] * 之前 02.php 面向接口开发,客户端还是知道了服务器端的所有类. * 现在想让客户端只知道一个类,就用工厂. */ header("Con ...
- 关于ADO.NET@SQL Server&SqlDataReader
先说基础的,说基础的明白了再深的也是一样的.SQL是关系型数据库,所以就决定了对其操作的时候ADO的一些类要相互联系,Connection 类Command对象(ExecuteReader()方法.E ...
- 微信公众平台消息接口API指南
简介 微信公众平台消息接口为开发者提供了一种新的消息处理方式.微信公众平台消息接口为开发者提供与用户进行消息交互的能力.对于成功接入消息接口的微信公众账号,当用户发消息给公众号,微信公众平台服务器会使 ...