洛谷5038 [SCOI2012]奇怪的游戏(二分+网络流+判断奇偶)
寒假的时候就听过这个题。但是一直没有写。
qwq
首先,我们发现题目中的图是个网格图,然后每次可以将相邻两个格子加一。
很容易就想到是黑白染色。那么每次操作,就相当于同时操作一个白点,一个黑点。
我们会发现,这样其实到最终局面的时候,黑点和白点所加的差是相等的,也就是说,我们假设黑点的个数是\(num1\),权值和是\(sum1\),白点的个数是\(num2\),权值和是\(sum2\)。若最终局面的数字是\(x\)
\]
那么$$x = \frac{sum1-sum2}{num1-num2}$$
比较容易发现的是,如果\(num1!=num2\),那我们可以直接求出来这个\(x\),然后\(check\)一下就好。
那如果\(num1==num2\)呢?
由于两个格子的数目是一样的,所以,只要我们能到达\(x\),我们也一定可以通过一些+的操作,使得局面能到达\(x+1\)。那么这时候就可以直接二分+\(check\)了。
那么该如果check呢?
其实也比较简单。
\(S\rightarrow白点,黑点\rightarrow T\)
流量是他们的权值和二分的值的差,然后黑白点之间的流量是\(inf\),直接跑\(dinic\)即可。
记得最终的\(ans\)是操作次数,不是最小的合成的数。
#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
#include<queue>
#include<map>
#include<set>
#define mk make_pair
#define ll long long
#define int long long
using namespace std;
inline int read()
{
int x=0,f=1;char ch=getchar();
while (!isdigit(ch)) {if (ch=='-') f=-1;ch=getchar();}
while (isdigit(ch)) {x=(x<<1)+(x<<3)+ch-'0';ch=getchar();}
return x*f;
}
const int maxn = 5010;
const int maxm = 2e6+1e2;
const int inf = 2e18;
int point[maxn],nxt[maxm],to[maxm];
int val[maxm],cnt=1,n,m;
int a[100][100];
int col[100][100];
void addedge(int x,int y,int w)
{
nxt[++cnt]=point[x];
to[cnt]=y;
point[x]=cnt;
val[cnt]=w;
}
void insert(int x,int y,int w)
{
addedge(x,y,w);
addedge(y,x,0);
}
int h[maxn];
int s,t;
queue<int> q;
int mx;
int tmp;
bool bfs(int s)
{
memset(h,-1,sizeof(h));
h[s]=0;
q.push(s);
while (!q.empty())
{
int x=q.front();
q.pop();
for (int i=point[x];i;i=nxt[i])
{
int p = to[i];
if (h[p]==-1 && val[i]>0)
{
q.push(p);
h[p]=h[x]+1;
}
}
}
if (h[t]==-1) return false;
return true;
}
int dfs(int x,int low)
{
if (x==t || low==0) return low;
int totflow=0;
for (int i=point[x];i;i=nxt[i])
{
int p = to[i];
if (h[p]==h[x]+1 && val[i]>0)
{
int tmp = dfs(p,min(low,val[i]));
val[i]-=tmp;
val[i^1]+=tmp;
totflow+=tmp;
low-=tmp;
if(low==0) return totflow;
}
}
if(low>0) h[x]=-1;
return totflow;
}
int dinic()
{
int ans=0;
while (bfs(s))
{
ans=ans+dfs(s,inf);
}
return ans;
}
void init()
{
cnt=1;
memset(point,0,sizeof(point));
}
int getnum(int x,int y)
{
return (x-1)*m+y;
}
int dx[5]={0,-1,0,1,0};
int dy[5]={0,0,-1,0,1};
bool check(int mid)
{
init();
if (mid<mx) return false;
int sum1=0,sum2=0;
for (int i=1;i<=n;i++)
for (int j=1;j<=m;j++)
{
if (col[i][j]==1) insert(s,getnum(i,j),mid-a[i][j]),sum1+=mid-a[i][j];
else insert(getnum(i,j),t,mid-a[i][j]),sum2+=mid-a[i][j];
//sum+=mid-a[i][j];
}
if (sum1!=sum2) return false;
for (int i=1;i<=n;i++)
{
for (int j=1;j<=m;j++)
{
for (int k=1;k<=4;k++)
{
int ii = i+dx[k];
int jj = j+dy[k];
if(ii<=0 || ii>n || jj<=0 || jj>m) continue;
if (col[i][j]==2) continue;
insert(getnum(i,j),getnum(ii,jj),inf);
}
}
}
tmp=dinic();
return (tmp==sum1);
}
signed main()
{
int T=read();
while (T--)
{
init();
n=read();m=read();
mx=0;
s=maxn-10;
t=s+1;
for (int i=1;i<=n;i++)
for (int j=1;j<=m;j++)
{
if ((i+j)&1)
col[i][j]=1;
else col[i][j]=2;
a[i][j]=read();
mx=max(a[i][j],mx);
}
int num1=0,num2=0;
int sum1=0,sum2=0;
for (int i=1;i<=n;i++)
{
for (int j=1;j<=m;j++)
{
if (col[i][j]==1)
{
num1++;
sum1+=a[i][j];
}
else
{
num2++;
sum2+=a[i][j];
}
}
}
int ans = -1;
int l=0,r=1e18;
//cout<<1<<endl;
if (num1==num2)
{
while (l<=r)
{
int mid = (l+r) >> 1;
if (check(mid)) ans=tmp,r=mid-1;
else l=mid+1;
//cout<<1<<endl;
}
}
else
{
int now = (sum1-sum2)/(num1-num2);
if (check(now)) ans=tmp;
}
cout<<ans<<"\n";
}
return 0;
}
洛谷5038 [SCOI2012]奇怪的游戏(二分+网络流+判断奇偶)的更多相关文章
- 洛谷$P5038\ [SCOI2012]$奇怪的游戏 二分+网络流
正解:二分+网络流 解题报告: 传送门$QwQ$ 这种什么,"同时增加",长得就挺网络流的$QwQ$?然后看到问至少加多少次,于是考虑加个二分呗?于是就大体确定了做题方向,用的网络 ...
- bzoj 2756 [SCOI2012]奇怪的游戏 二分+网络流
2756:[SCOI2012]奇怪的游戏 Time Limit: 40 Sec Memory Limit: 128 MBSubmit: 4926 Solved: 1362[Submit][Stat ...
- P5038 [SCOI2012]奇怪的游戏 二分+网络流
$ \color{#0066ff}{ 题目描述 }$ Blinker最近喜欢上一个奇怪的游戏. 这个游戏在一个 \(N \times M\) 的棋盘上玩,每个格子有一个数.每次\(Blinker\)会 ...
- BZOJ2756 [SCOI2012]奇怪的游戏 【网络流 + 二分】
题目 Blinker最近喜欢上一个奇怪的游戏. 这个游戏在一个 N*M 的棋盘上玩,每个格子有一个数.每次 Blinker 会选择两个相邻 的格子,并使这两个数都加上 1. 现在 Blinker 想知 ...
- 洛谷——P1747 好奇怪的游戏
P1747 好奇怪的游戏 题目背景 <爱与愁的故事第三弹·shopping>娱乐章. 调调口味来道水题. 题目描述 爱与愁大神坐在公交车上无聊,于是玩起了手机.一款奇怪的游戏进入了爱与愁大 ...
- 洛谷 P1747 好奇怪的游戏
P1747 好奇怪的游戏 题目背景 <爱与愁的故事第三弹·shopping>娱乐章. 调调口味来道水题. 题目描述 爱与愁大神坐在公交车上无聊,于是玩起了手机.一款奇怪的游戏进入了爱与愁大 ...
- BZOJ.2756.[SCOI2012]奇怪的游戏(二分 黑白染色 最大流ISAP)
题目链接 \(Description\) \(Solution\) 这种题当然要黑白染色.. 两种颜色的格子数可能相同,也可能差1.记\(n1/n2\)为黑/白格子数,\(s1/s2\)为黑/白格子权 ...
- [洛谷P1747]好奇怪的游戏
题目大意:有两匹马,马可以走"日",也可以像象走"田",求它走到(1,1)的步数. 题解:bfs 卡点:边界判断成了可以走到(0,y)或(x,0) C++ Co ...
- 【BZOJ 2756】[SCOI2012]奇怪的游戏 二分+最大流
这道题提醒我,要有将棋盘黑白染色的意识,尤其是看到相邻格子这样的条件的时候,然后就是要用到与其有关的性质与特点以体现其作用,这道题就是用到了黑格子与白格子之间的关系进行的,其出发点是每次一定会给一个黑 ...
随机推荐
- jdbc操作mysql(三):利用注解封装
案例五:利用注解封装 重复步骤 我们使用jdbc操作mysql时发现,操作不同表中数据,所写的方法基本相同:比如我们根据id向用户表添加数据,根据id删除商品表的数据,或者查询所有数据并用list集合 ...
- opencv入门系列教学(七)改变颜色空间、提取彩色对象
0.序言 之前的博客里我们介绍了opencv在图像上的基本操作,下面我们来进行稍微深入一点的介绍,从这里开始我们可以发现opencv库能给我们带来的更多更有趣的功能.从现在开始,我们将逐步深入了解 ...
- jquery mobile常用的data-role类型
data-role参数表: page 页面容器,其内部的mobile元素将会继承这个容器上所设置的属性 header 页面标题容器,这个容器内部可以包含文字.返回按钮.功能按钮等 ...
- iNeuOS工业互联平台,增加OPC UA驱动,同步和订阅方式读取数据
目 录 1. 概述... 1 2. 平台演示... 2 3. OPC UA应用效果... 2 1. 概述 最近的项目,用户需要使用OPC UA读取数据,通 ...
- 集合框架3-Arrays 类
Arrays 和 Collections是分别操作数组和集合的两个工具类.今天就来对 Arrays 中的内容作个总结. 一.Arrays 类概述 Arrays 类位于 java.util包中.Arra ...
- B. 2194: 快速傅立叶之二解题报告
$$\begin{eqnarray}&c[k] = \sum_{i}^{n}a[i]b[i-k] \\&c[k] = \sum_{i}^{n}a[n-i]b[i-k] (倒序保存a) ...
- token过期时间
token有两个有效期,在config/jwt.php下面 # ./config/jwt.php'ttl' => env('JWT_TTL', 600), ##600min=10h 'refre ...
- leetcode数组典型题目小结
数组与矩阵 数组与矩阵的基本知识: 1.数组:数组是在程序设计中,为了处理方便, 把具有相同类型的若干元素按有序的形式组织起来的一种形式. 首先,数组会利用索引来记录每个元素在数组中的位置,且在大多数 ...
- Apache配置与应用
目录: 一.基于域名的虚拟主机 二.基于IP地址的虚拟主机 三.基于端口的虚拟主机 四.Apache连接保持 五.构建Web虚拟目录与用户授权限制 六.Apache日志分割 七.AWStats 分析系 ...
- 配置IIS Express 允许外部访问
修改applicationhost.config 配置允许外部访问 操作步骤: 1. 查看本机IP地址记录IP地址,例如:10.1.20.138 2. 如下图,找到要发布的站点的名称 记录站点的名称, ...