题解 P5038 [SCOI2012]奇怪的游戏
题解
做这题之前,做了一道叫星际战争的题,很容易想到二分 \(+\) 网络流,那么二分啥呢?
我们先推一下式子,因为是对相邻格子加数,那么可以联想到黑白染色类问题。
设有黑色格子 \(B\) 个,其格子中初始数的和为 \(b\),白色格子同理,个数为 \(W\) 个,初始权值和为 \(w\) 个,最后变成的同一个数为 \(num\)。
可以得出 \(B×num-b=W×num-w\) 化简得 \(num=\frac{b-w}{B-W}\)。
首先对于化简式,其必要条件是 \(B\neq W\),那么我们分两种情况讨论。
对于 \(B\neq W\) 的情况,那么我们可以直接算出 \(num\) ,但是我们要检验其是否合法。
对于 \(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]奇怪的游戏的更多相关文章
- [题目] Luogu P5038 [SCOI2012]奇怪的游戏
学习资料 -----1----- -----2----- P5038 [SCOI2012]奇怪的游戏 一道甚神但没用到高深模型的题 思路 没思路,看题解吧 代码 #include <iostre ...
- P5038 [SCOI2012]奇怪的游戏 二分+网络流
$ \color{#0066ff}{ 题目描述 }$ Blinker最近喜欢上一个奇怪的游戏. 这个游戏在一个 \(N \times M\) 的棋盘上玩,每个格子有一个数.每次\(Blinker\)会 ...
- P5038 [SCOI2012]奇怪的游戏
题目链接 题意分析 首先我们需要求的是统一以后的值\(x\) 并且一般的棋盘操作我们都需要黑白染色 那么对于棋盘格子是偶数的情况的话 答案是存在单调性的 因为如果统一之后 两两搭配还是可以再加一个的 ...
- 洛谷$P5038\ [SCOI2012]$奇怪的游戏 二分+网络流
正解:二分+网络流 解题报告: 传送门$QwQ$ 这种什么,"同时增加",长得就挺网络流的$QwQ$?然后看到问至少加多少次,于是考虑加个二分呗?于是就大体确定了做题方向,用的网络 ...
- BZOJ 2756: [SCOI2012]奇怪的游戏 网络流/二分
2756: [SCOI2012]奇怪的游戏 Time Limit: 40 Sec Memory Limit: 128 MBSubmit: 1594 Solved: 396[Submit][Stat ...
- bzoj2756: [SCOI2012]奇怪的游戏(网络流+分情况)
2756: [SCOI2012]奇怪的游戏 题目:传送门 题解: 发现做不出来的大难题一点一个网络流 %大佬 首先黑白染色(原来是套路...)染色之后就可以保证每次操作都一定会使黑白各一个各自的值加1 ...
- BZOJ 2756: [SCOI2012]奇怪的游戏 [最大流 二分]
2756: [SCOI2012]奇怪的游戏 Time Limit: 40 Sec Memory Limit: 128 MBSubmit: 3352 Solved: 919[Submit][Stat ...
- Bzoj2756 [SCOI2012]奇怪的游戏
2756: [SCOI2012]奇怪的游戏 Time Limit: 40 Sec Memory Limit: 128 MBSubmit: 3220 Solved: 886 Description ...
- bzoj 2756 [SCOI2012]奇怪的游戏 二分+网络流
2756:[SCOI2012]奇怪的游戏 Time Limit: 40 Sec Memory Limit: 128 MBSubmit: 4926 Solved: 1362[Submit][Stat ...
随机推荐
- WPF教程九:理解WPF中的对象资源
在WPF中,所有继承自FrameworkElement的元素都包含一个Resources属性,这个属性就是我们这篇要讲的资源. 这一篇讲解的资源是不是上一篇的程序集资源(那个是在编译过程中打包到程序集 ...
- 新旧图号(图幅号)转换/计算/检查,经纬度转换计算,C#代码
图号(图幅号):地图图号是指为便于使用和管理,按照一定方法将各分幅地图进行的编号. 经常用到图号,但是在网上一直没有找到一个完整的图号转换程序,因此自己写了一个图号处理的库,分享出来.如有错误请指正. ...
- Vue数据双向绑定不起作用、Vue如何正确的手动添加json数据、Vue视图层不刷新、手动刷新视图层
Vue.set(obj,"key","value") 如果接收到来自服务器的消息时,我们需要对其进性进一步处理 我们想当然的会直接将数据添加进json 像这样: ...
- vue(19)嵌套路由
嵌套路由 有时候在路由中,主要的部分是相同的,但是下面可能是不同的.比如访问首页,里面有新闻类的/home/news,还有信息类的/home/message.这时候就需要使用到嵌套路由.项目结构如下: ...
- P5311 [Ynoi2011] 成都七中
P5311 [Ynoi2011] 成都七中 题意 给你一棵 \(n\) 个节点的树,每个节点有一种颜色,有 \(m\) 次查询操作. 查询操作给定参数 \(l\ r\ x\),需输出: 将树中编号在 ...
- TS中 使用deprecated 实现对方法的迭代弃用
在日常开发中,我们会定义大量方法函数来提供给业务调用,可随着时间与业务的推进, 有些方法可能不切合当下需求, 或将被逐步废弃并替换到新的方法中, 例如 框架中 部分生命周期的废弃. 此时作为开发者就很 ...
- jvm源码解读--01 jvm加载java/lang/object过程
现在做一下记录,这个看了两天,看的过程发现了很多c++的高级特性,没接触过,还得慢慢撸,禁止很慢 那么现在开始 吧 先打两个断点 java.c:351 JavaMain(void * _args) { ...
- 3分钟搭建一个网站?腾讯云Serverless开发体验
作为一个开发者,应该都能理解一个网站从开发到上线,要经过很多繁琐的步骤. 编写代码,部署应用,部署数据库,申请域名,申请SSL证书,域名备案,到最终上线起码要几天时间. 作为一个不精通代码的业务玩家, ...
- 大数据学习(22)—— ZooKeeper能做些什么
官网上已经给出了zk的几种典型应用场景,原话是这么说的: It exposes a simple set of primitives that distributed applications can ...
- Lambda--Optional、Collectors高级进阶方法
Lambda--Collectors.optional高级使用 偶然看到了同事groupingBy用法,然后百度衍生出了optional,collectors,map等各种用法.突然发现自己之前写的代 ...