题解

题目

做这题之前,做了一道叫星际战争的题,很容易想到二分 \(+\) 网络流,那么二分啥呢?

我们先推一下式子,因为是对相邻格子加数,那么可以联想到黑白染色类问题。

设有黑色格子 \(B\) 个,其格子中初始数的和为 \(b\),白色格子同理,个数为 \(W\) 个,初始权值和为 \(w\) 个,最后变成的同一个数为 \(num\)。

可以得出 \(B×num-b=W×num-w\) 化简得 \(num=\frac{b-w}{B-W}\)。

首先对于化简式,其必要条件是 \(B\neq W\),那么我们分两种情况讨论。

  1. 对于 \(B\neq W\) 的情况,那么我们可以直接算出 \(num\) ,但是我们要检验其是否合法。

  2. 对于 \(B=W\) 的情况,首先我们要保证 \(b=w\) 否则直接无解。然后,对于一个合法的 \(num\),\(num+1\) 也是合法的,因为 \(B=W\) 所以在 \(n,m\) 中一定有一个是偶数。于是我们就可以二分求解,找到临界的 \(num\) 就是答案。

这个 \(check()\) 我们可以用最大流求解,对每个黑格子,我们分别向源点 \(s\),权值为 \(num-w_i\),它能到的白格子,权值为 \(INF\) 连边,对每个白格子,我们向汇点 \(t\) 连边,权值为 \(num-w_i\)。

对这张图跑最大流,如果最大流等于 \(B×num-b\) 那么说明这张图跑满了,说明答案正确。

Code:

\(AC\kern 0.4em CODE:\)

#include<bits/stdc++.h>
#define ri register signed
#define p(i) ++i
using namespace std;
namespace IO{
char buf[1<<21],*p1=buf,*p2=buf;
#define gc() p1==p2&&(p2=(p1=buf)+fread(buf,1,1<<21,stdin),p1==p2)?EOF:*p1++
inline int read() {
ri x=0,f=1;char ch=gc();
while(ch<'0'||ch>'9') {if (ch=='-') f=-1;ch=gc();}
while(ch>='0'&&ch<='9') {x=(x<<1)+(x<<3)+(ch^48);ch=gc();}
return x*f;
}
}
using IO::read;
namespace nanfeng{
#define int long long//懒人必备
#define cmax(x,y) ((x)>(y)?(x):(y))
#define cmin(x,y) ((x)>(y)?(y):(x))
#define FI FILE *IN
#define FO FILE *OUT
#undef bool
static const int N=45;
int ch[N][N],id[N][N],n,m,T,tot=1,B,b,W,w,mx;
namespace NetworkFlows{
#define jud(i,j) ((i)&&(j)&&(i)<=n&&(j)<=m)
static const int INF=1e18+7;
int first[N*N],dep[N*N],cur[N*N],que[N*N],t=2,sw,s=1,et;
signed dx[5]={0,1,0,-1,0},dy[5]={0,0,1,0,-1};
struct edge{int v,nxt,w;}e[N*N*5];
inline void add(int u,int v,int w) {
e[t].v=v,e[t].w=w;
e[t].nxt=first[u];
first[u]=t++;
}
inline void init() {et=n*m+2,t=2,sw=0;memset(first,0,sizeof(first));}
inline void build(int w) {
for (ri i(1);i<=n;p(i)) {
for (ri j(1);j<=m;p(j)) {
if (!((i+j)%2)) {
sw+=w-ch[i][j];
add(s,id[i][j],w-ch[i][j]);
add(id[i][j],s,0);
for (ri d(1);d<=4;p(d)) {
if (jud(i+dx[d],j+dy[d])) add(id[i][j],id[i+dx[d]][j+dy[d]],INF),add(id[i+dx[d]][j+dy[d]],id[i][j],0);
}
} else add(id[i][j],et,w-ch[i][j]),add(et,id[i][j],0);
}
}
}
inline bool bfs(int s,int t) {
memset(dep,0,sizeof(dep));
ri hd=1,tl=0;
dep[que[p(tl)]=s]=1;
cur[s]=first[s];
while(hd<=tl) {
s=que[hd++];
for (ri i(first[s]),v;i;i=e[i].nxt) {
if (e[i].w&&!dep[v=e[i].v]) {
dep[v]=dep[s]+1;
cur[que[p(tl)]=v]=first[v];
if (v==t) return 1;
}
}
}
return 0;
}
int dfs(int x,int flow) {
if (x==et||!flow) return flow;
int rst=flow;
for (ri i(cur[x]),v;i&&rst;i=e[i].nxt) {
if (e[i].w&&dep[v=e[i].v]==dep[x]+1) {
register int k;
if (!(k=dfs(v,cmin(e[i].w,rst)))) dep[v]=0;
e[i].w-=k,e[i^1].w+=k,rst-=k;
}
cur[x]=i;
}
return flow-rst;
}
inline int dinic() {
int res=0;
while(bfs(s,et)) res+=dfs(s,INF);
return res;
}
}
inline bool check(int w) {
NetworkFlows::init();
NetworkFlows::build(w);
return NetworkFlows::dinic()==NetworkFlows::sw;
}
inline void init() {mx=B=b=W=w=0,tot=1;}
inline int main() {
// FI=freopen("nanfeng.in","r",stdin);
// FO=freopen("nanfeng.out","w",stdout);
T=read();
for (ri z(1);z<=T;p(z)) {
init();
n=read(),m=read();
for (ri i(1);i<=n;p(i)) {
for (ri j(1);j<=m;p(j)) {
ch[i][j]=read(),id[i][j]=p(tot);
mx=cmax(mx,ch[i][j]);
if (!((i+j)%2)) p(B),b+=ch[i][j];
else p(W),w+=ch[i][j];
}
}
if (B!=W) {
int num=(b-w)/(B-W);
if (num>=mx&&check(num)) printf("%lld\n",B*num-b);
else puts("-1");
} else {
if (b!=w) {puts("-1");continue;}
int l=mx,r=1e11,res=-1;
while(l<=r) {
int mid((l+r)>>1);
if (check(mid)) r=mid-1,res=mid;
else l=mid+1;
}
printf("%lld\n",res==-1?-1:B*res-b);
}
}
return 0;
}
#undef int
}
int main() {return nanfeng::main();}

题解 P5038 [SCOI2012]奇怪的游戏的更多相关文章

  1. [题目] Luogu P5038 [SCOI2012]奇怪的游戏

    学习资料 -----1----- -----2----- P5038 [SCOI2012]奇怪的游戏 一道甚神但没用到高深模型的题 思路 没思路,看题解吧 代码 #include <iostre ...

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

    $ \color{#0066ff}{ 题目描述 }$ Blinker最近喜欢上一个奇怪的游戏. 这个游戏在一个 \(N \times M\) 的棋盘上玩,每个格子有一个数.每次\(Blinker\)会 ...

  3. P5038 [SCOI2012]奇怪的游戏

    题目链接 题意分析 首先我们需要求的是统一以后的值\(x\) 并且一般的棋盘操作我们都需要黑白染色 那么对于棋盘格子是偶数的情况的话 答案是存在单调性的 因为如果统一之后 两两搭配还是可以再加一个的 ...

  4. 洛谷$P5038\ [SCOI2012]$奇怪的游戏 二分+网络流

    正解:二分+网络流 解题报告: 传送门$QwQ$ 这种什么,"同时增加",长得就挺网络流的$QwQ$?然后看到问至少加多少次,于是考虑加个二分呗?于是就大体确定了做题方向,用的网络 ...

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

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

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

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

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

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

  8. Bzoj2756 [SCOI2012]奇怪的游戏

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

  9. bzoj 2756 [SCOI2012]奇怪的游戏 二分+网络流

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

随机推荐

  1. buu [V&N2020 公开赛]strangeCpp

    拖入ida,静态调试一下,本来想动调的,发现一直缺dll.没办法,只能头铁,静态 找到主函数,然后并没有发现什么,找了半天,没结果,后面也是看了大佬wp,才找到解决方式,感觉这种只能通过动调来找到关键 ...

  2. git研究详解(官网文档)及总结

    前言:git作为新一代的版本控制软件,说实话比svn好用多了,个人见解,关于git的详细介绍及研究,我推荐三个地方 1.git官网上的文档(推荐UC浏览器,比火狐多个英文翻译的功能) 地址为:http ...

  3. 递推算法,AI衍生

    引言 最近在刷leetcode算法题的时候,51题很有意思: 题目是这样的: n 皇后问题 研究的是如何将 n 个皇后放置在 n×n 的棋盘上,并且使皇后彼此之间不能相互攻击.给你一个整数 n ,返回 ...

  4. VisibleDeprecationWarning , Creating an ndarray from ragged nested sequences... 警告怎么办

    我不是完美主义,但是至少,我在做实验的时候不能容忍有 warning 的出现. 今天使用 tensorflow.keras.datasets中的 imdb 数据集,使用 imdb.load_data( ...

  5. ESP32低功耗模式

    1.ESP32 系列芯片提供三种可配置的睡眠模式,针对这些睡眠模式,我们提供了了多种低功耗解决方案,用户可以结合具体需求选择睡眠模式并进行配置.三种睡眠模式如下: Modem-sleep 模式:CPU ...

  6. HAL库直流电机编码测速(L298N驱动)笔记

    主函数开始后的处理流程: 1.外设初始化:HAL_Init() 2.系统时钟配置 RCC振荡器初始化:HAL_RCC_OsConfig() RCC时钟初始化:HAL_RCC_ClockConfig() ...

  7. Python使用笔记005-文件操作(二)

    1.1 打开文件模式 # r r+ r+读是没问题的,写的话,会覆盖原来的内容,文件不存在时会报错# w w+ w+用来新的文件没问题,旧的文件会覆盖原来的内容# a a+ a+写是追加,读不到是因为 ...

  8. urllib3使用池管理发送请求和requests常用方法的基本使用+session使用

    使用urllib3的池管理器 urllib3是在urllib进行更加深入的改进,最大的好处就是在urllib的基础上添加了池管理,以至于我们不需要再去注意我们需要由那个链接去发送请求,而只需要获取到链 ...

  9. Ubuntu18.04 安装opensips,实现局域网内sip语音视频通话

    Ubuntu18.04直接安装opensips 本人实践亲测有效,用docker安装opensips尝试多次均无法连接mysql数据库,故舍弃,直接在主机上安装opensips 部分内容参考自:htt ...

  10. Python Unittest简明教程

    1 概述 单元测试框架是一种软件测试方法,通过来测试源代码中的各个单元,例如类,方法等,以确定它们是否符合要求.直观上来说,可以将单元视为最小的可测试部分.单元测试是程序员在开发过程中创建的短代码片段 ...