Description

Blinker最近喜欢上一个奇怪的游戏。
这个游戏在一个 N*M 的棋盘上玩,每个格子有一个数。每次 Blinker 会选择两个相邻
的格子,并使这两个数都加上 1。
现在 Blinker 想知道最少多少次能使棋盘上的数都变成同一个数,如果永远不能变成同
一个数则输出-1。

Input

输入的第一行是一个整数T,表示输入数据有T轮游戏组成。
每轮游戏的第一行有两个整数N和M, 分别代表棋盘的行数和列数。
接下来有N行,每行 M个数。

Output

对于每个游戏输出最少能使游戏结束的次数,如果永远不能变成同一个数则输出-1。

Sample Input

2
2 2
1 2
2 3
3 3
1 2 3
2 3 4
4 3 2

Sample Output

2
-1

HINT

【数据范围】

对于30%的数据,保证  T<=10,1<=N,M<=8
对于100%的数据,保证  T<=10,1<=N,M<=40,所有数为正整数且小于1000000000

Solution

emmm这个题真的有毒啊……INF改成8e8才过……最慢的点跑2s……幸亏BZOJ算总时间
首先看到这个数据范围十有八九网络流没跑了……
先将格子黑白染色,考虑一通操作成功后每个格子的数都是v,若:
1、n*m是奇数。设黑白格子个数为num[0/1],和为sum[0/1],那么$num0*x-sum0=num1*x-sum1$,因为每次操作肯定会给一个黑点和一个白点加。
2、n*m是偶数。如果最终答案是v满足的话,那么显然最终答案是v+1肯定也是可以满足的。这个就可以二分了。
现在的问题转换为判断是否可行。
s-black,容量为v-a[i][j]
white-e,容量为v-a[i][j]
black-white, 容量为INF。
怎么理解呢?把INF边看成给两端的数各自加一,而INF边连着的两端的边又限制了INF边的使用次数。

Code

 #include<iostream>
#include<cstring>
#include<cstdio>
#include<queue>
#define N (2001)
#define INF (8000000000LL)
#define LL long long
using namespace std; struct Edge{LL to,next,flow;}edge[N<<];
LL T,n,m,s,e=,Depth[N],num[],sum[],maxn,cnt;
LL head[N],num_edge,a[N][N],col[N][N],id[N][N];
LL dx[]={,,-,,},dy[]={,,,,-};
queue<LL>q; void add(LL u,LL v,LL l)
{
edge[++num_edge].to=v;
edge[num_edge].next=head[u];
edge[num_edge].flow=l;
head[u]=num_edge;
} LL Dfs(LL x,LL low)
{
if (x==e || low==) return low;
LL f=,Min=;
for (int i=head[x]; i; i=edge[i].next)
if (edge[i].flow && Depth[edge[i].to]==Depth[x]+ && (Min=Dfs(edge[i].to,min(low,edge[i].flow))))
{
edge[i].flow-=Min;
edge[((i-)^)+].flow+=Min;
f+=Min; low-=Min;
if (!low) break;
}
if (!f) Depth[x]=-;
return f;
} bool Bfs(LL s,LL e)
{
memset(Depth,,sizeof(Depth));
Depth[s]=; q.push(s);
while (!q.empty())
{
LL x=q.front(); q.pop();
for (int i=head[x]; i; i=edge[i].next)
if (!Depth[edge[i].to] && edge[i].flow)
{
Depth[edge[i].to]=Depth[x]+;
q.push(edge[i].to);
}
}
return (Depth[e]!=);
} bool Dinic(LL s,LL e)
{
while (Bfs(s,e)) Dfs(s,INF);
for (int i=head[s]; i; i=edge[i].next)
if (edge[i].flow!=) return false;
for (int i=head[e]; i; i=edge[i].next)
if (edge[((i-)^)+].flow!=) return false;
return true;
} bool check(LL v)
{
memset(head,,sizeof(head)); num_edge=;
memset(edge,,sizeof(edge));
for (int i=; i<=n; ++i)
for (int j=; j<=m; ++j)
{
if (col[i][j]) add(s,id[i][j],v-a[i][j]),add(id[i][j],s,);
else add(id[i][j],e,v-a[i][j]),add(e,id[i][j],);
if (!col[i][j]) continue;
for (int k=; k<=; ++k)
{
LL x=i+dx[k], y=j+dy[k];
if (x< || x>n || y< || y>m) continue;
add(id[i][j],id[x][y],INF),add(id[x][y],id[i][j],);
}
}
return Dinic(s,e);
} int main()
{
scanf("%lld",&T);
while (T--)
{
scanf("%lld%lld",&n,&m);
sum[]=sum[]=maxn=cnt=;
if (n*m%) num[]=n*m/+,num[]=n*m/;
else num[]=num[]=n*m/;
for (int i=; i<=n; ++i)
for (int j=; j<=m; ++j)
{
scanf("%lld",&a[i][j]);
id[i][j]=++cnt;
maxn=max(maxn,a[i][j]);
if (i%==j%) col[i][j]=;
sum[col[i][j]]+=a[i][j];
}
if (n*m%)
{
LL v=(sum[]-sum[])/(num[]-num[]);
if (check(v)) printf("%lld\n",(v*n*m-sum[]-sum[])/);
else printf("-1\n");
}
else
{
LL l=maxn, r=INF, ans=-;
while (l<=r)
{
LL mid=(l+r)/;
if (check(mid)) r=mid-,ans=mid;
else l=mid+;
}
if (ans==-) printf("-1\n");
else printf("%lld\n",(ans*n*m-sum[]-sum[])/);
}
}
}

BZOJ2756:[SCOI2012]奇怪的游戏(最大流,二分)的更多相关文章

  1. BZOJ 2756: [SCOI2012]奇怪的游戏 [最大流 二分]

    2756: [SCOI2012]奇怪的游戏 Time Limit: 40 Sec  Memory Limit: 128 MBSubmit: 3352  Solved: 919[Submit][Stat ...

  2. BZOJ2756 [SCOI2012]奇怪的游戏 【网络流 + 二分】

    题目 Blinker最近喜欢上一个奇怪的游戏. 这个游戏在一个 N*M 的棋盘上玩,每个格子有一个数.每次 Blinker 会选择两个相邻 的格子,并使这两个数都加上 1. 现在 Blinker 想知 ...

  3. BZOJ2756 [SCOI2012]奇怪的游戏 最大流

    好久没有写博客了.不过这个博客也没有多少人看 最近在写网络流,为了加深理解,来写一两篇题解. 对整个棋盘进行黑白染色以后可以发现,一次操作就是让二分图的两个点的值分别 \(+1\). 这样,我们就可以 ...

  4. 【BZOJ-2756】奇怪的游戏 最大流 + 分类讨论 + 二分

    2756: [SCOI2012]奇怪的游戏 Time Limit: 40 Sec  Memory Limit: 128 MBSubmit: 2925  Solved: 792[Submit][Stat ...

  5. Bzoj2756 [SCOI2012]奇怪的游戏

    2756: [SCOI2012]奇怪的游戏 Time Limit: 40 Sec  Memory Limit: 128 MBSubmit: 3220  Solved: 886 Description ...

  6. bzoj2756: [SCOI2012]奇怪的游戏(网络流+分情况)

    2756: [SCOI2012]奇怪的游戏 题目:传送门 题解: 发现做不出来的大难题一点一个网络流 %大佬 首先黑白染色(原来是套路...)染色之后就可以保证每次操作都一定会使黑白各一个各自的值加1 ...

  7. BZOJ 2756 SCOI2012 奇怪的游戏 最大流

    题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=2756 Description Blinker最近喜欢上一个奇怪的游戏. 这个游戏在一个 N ...

  8. BZOJ2756 SCOI2012奇怪的游戏(二分答案+最大流)

    由数据范围容易想到网络流.由于操作只是对于棋盘上相邻两格,容易想到给其黑白染色. 假设已经知道最后要变成什么数.那么给黑白点之间连边,其流量则表示同时增加的次数,再用源汇给其限流为需要增加的数即可. ...

  9. BZOJ 2756: [SCOI2012]奇怪的游戏 网络流/二分

    2756: [SCOI2012]奇怪的游戏 Time Limit: 40 Sec  Memory Limit: 128 MBSubmit: 1594  Solved: 396[Submit][Stat ...

随机推荐

  1. Golang教程:循环语句

    循环语句用于重复执行一段代码. for 语句是 Go 中唯一的循环语句.Go 没有提供其他语言(如 C)中的 while 和 do while 语句. for 语句语法 for 语句的语法如下: fo ...

  2. 点击checkbox全选,其它被选中,再点击取消

    <input type="checkbox" value="" id="checkall" name="" siz ...

  3. C# 委托,事件, 异步

    委托 ​ 委托可以说是方法的集合,里面放着方法列表,当调用的时候就会调用全部的方法列表 ​ 个人理解 : 当声明和创建时委托时, 它是个 对象 当使用委托,调用方法时,它是个 方法 声明委托类型 de ...

  4. mongodb基本指令

    MongoDB基本命令用成功启动MongoDB后,再打开一个命令行窗口输入mongo,就可以进行数据库的一些操作. 输入help可以看到基本操作命令: show dbs:显示数据库列表 show co ...

  5. 三:Mybatis知识整理(3)

    一:mybatis中模糊查询的方法: 1.直接传参法:在java传参时进行拼接 -- %keyword% 2.mysql内置函数:concart('%',#{keyword},'%') -- 拼接sq ...

  6. java——程序的导出与导入

    导出: 选择项目,右击选择 最下面的properties——Resource——Location,就是你的项目所在地, 找到文件所在,拷贝到你的U盘中(或者直接点击项目直接拖到桌面)完成复制 导入: ...

  7. <td>标签scope属性

    HTML <td> 标签的 scope 属性 HTML <td> 标签 实例 下面的例子把两个 th 元素标识为列的表头,把两个 td 元素标识为行的表头: <table ...

  8. SQL2008使用with求余额表,流水账方式

    1.先准备数据,将要求余额的表数据插入临时表 SELECT Serial, VoucherNum, SubjectNum, SubjectName, Direction, Amount INTO #T ...

  9. 深入理解jQuery插件开发总结(四)

    jQuery插件开发模式 软件开发过程中是需要一定的设计模式来指导开发的,有了模式,我们就能更好地组织我们的代码,并且从这些前人总结出来的模式中学到很多好的实践. 根据<jQuery高级编程&g ...

  10. WHILE (Transact-SQL)

    ---循环 declare @n int declare @rowcount int declare @name varchar(50) create table #temp ( id int ide ...