Link:https://www.lydsy.com/JudgeOnline/problem.php?id=2756

Algorithm:

比较新颖的题目

首先发现是对矩阵中相邻两数进行操作    <----->    想到黑白染色

于是Delta(BlackSum)=Delta(WhiteSum)

由于最后要变成同一个数X,

那么在BlackNum=WhiteNum时,

1、如果WhiteSum!=BlackSum,显然无解

2、如果WhiteSum==BlackSum时,由于矩阵能被1*2的矩形完全覆盖,那么X是否满足要求是具有单调性

于是我们二分X,进行判断即可

在WhiteNum!=BlackNum时,

显然可以得到X∗WhiteNum−WhiteSum=X∗BlackNum−BlackSum

移项后显然可以化简出X=(BlackSum−WhiteSum)  /   (BlackNum−WhiteNum)

验证这个X即可

最大问题,如何判断X是否可行,

一开始得出的性质:Delta(BlackSum)=Delta(WhiteSum),符合流守恒性(流入和流出量相同)

于是我们进行可以将黑点看作一边,而将白点看作另外一边,网络流建图:

S-->白点,CAP为X-val[i][j]

黑点-->T,CAP为X-val[i][j]

相邻的白点-->黑点,CAP为INF

判断能否满流即可

Code:

#include <bits/stdc++.h>

using namespace std;
#define pos(x,y) (x-1)*m+y
typedef pair<int,int> P;
typedef long long ll;
const int MAXN=;
const ll INF=1ll<<; ll test,n,m,col[MAXN][MAXN],dat[MAXN][MAXN],iter[MAXN*MAXN],level[MAXN*MAXN];
ll sum1,sum2,cnt1,cnt2,mx,S,T; struct edge
{
ll to,cap,rev;
};
vector<edge> G[MAXN*MAXN]; void add_edge(int from,int to,ll cap)
{
G[from].push_back(edge{to,cap,G[to].size()});
G[to].push_back(edge{from,,G[from].size()-});
} bool bfs()
{
memset(level,-,sizeof(level));
queue<int> que;que.push(S); level[S]=;
while(!que.empty())
{
int t=que.front();que.pop();
for(int i=;i<G[t].size();i++)
{
edge e=G[t][i];
if(e.cap> && level[e.to]==-)
{
level[e.to]=level[t]+;
que.push(e.to);
}
}
}
} ll dfs(int cur,int T,ll f)
{
if(cur==T) return f; for(ll &i=iter[cur];i<G[cur].size();i++)
{
edge &e=G[cur][i];
if(level[e.to]==level[cur]+ && e.cap>)
{
ll d=dfs(e.to,T,min(f,e.cap));
if(d>)
{
e.cap-=d;
G[e.to][e.rev].cap+=d;
return d;
}
}
}
return ;
} ll dinic()
{
ll ret=;
while(true)
{
memset(iter,,sizeof(iter));
bfs();if(level[T]==-) break;
ll f;
while((f=dfs(S,T,INF))>) ret+=f; //这里括号不能排错啊
}
return ret;
} int dx[]={,,,-};
int dy[]={,-,,}; bool check(ll tar)
{
S=;T=n*m+;ll ret=;
for(int i=;i<MAXN*MAXN;i++) G[i].clear();
for(int i=;i<=n;i++) //建图
for(int j=;j<=m;j++)
{
if(col[i][j]) add_edge(S,pos(i,j),tar-dat[i][j]),ret+=tar-dat[i][j];
else {add_edge(pos(i,j),T,tar-dat[i][j]);continue;}
for(int k=;k<;k++)
{
int fx=i+dx[k],fy=j+dy[k];
if(fx>= && fx<=n && fy>= && fy<=m)
add_edge(pos(i,j),pos(fx,fy),INF);
}
} ll MAXFLOW=dinic(); //dinic
return (ret==MAXFLOW);
} int main()
{
cin >> test;
while(test--)
{
cin >> n >> m;mx=;
cnt1=cnt2=sum1=sum2=;
for(int i=;i<=n;i++)
for(int j=;j<=m;j++)
{
cin >> dat[i][j];
col[i][j]=(i+j)&;mx=max(mx,dat[i][j]);
if(col[i][j]) sum1+=dat[i][j],cnt1++;
else sum2+=dat[i][j],cnt2++;
} if(cnt1!=cnt2)
{
ll tar=(sum1-sum2)/(cnt1-cnt2);
if(tar<mx){puts("-1");continue;} if(check(tar))
{
cout << tar*cnt1-sum1 << endl;
continue;
}
puts("-1");
}
else
{
if(sum1!=sum2){puts("-1");continue;} ll l=mx,r=INF;
while(l<=r)
{
ll mid=(l+r)>>;
if(check(mid)) r=mid-;
else l=mid+;
}
if(r==INF) puts("-1"); //如果到INF都无解,则说明就是无解,要特判,但BZOJ上数据比较水(黄学长的标程是错的)
else cout << 1ll*(l*cnt1-sum1) << endl;
}
}
return ;
}

Review:

1、看到这种对相邻格子同时操作的题目,想到黑白染色

2、求极值,观察值的可行性是否具有单调性,从而能否二分

3、当数据能分为两组,且具有流守恒性时

考虑使用网络流

4、调试:

(1)有多个括号要留心每个括号的位置啊,多看几遍

while((f=dfs(S,T,INF))>)

这里的>0一开始放到第2个括号里了,结果f每次就都是0或1了……囧,调了1h

(2)像网络流这样执行过一边就会对原数组产生影响的模块,

执行过一次后想调试/再次调用时,不能再执行一遍,将第一次结果存储即可

(3)由于模板中大部分数据都是int,如果有long long时要特别注意修改原模板中的int

惨案:

void add_edge(int from,int to,ll cap)

[BZOJ 2756] 奇怪的游戏的更多相关文章

  1. bzoj 2756奇怪的游戏

    2756: [SCOI2012]奇怪的游戏 Time Limit: 40 Sec  Memory Limit: 128 MB Description Blinke 最近喜欢上一个奇怪的游戏. 这个游戏 ...

  2. BZOJ 2756 奇怪的游戏(最大流)

    题目链接:http://61.187.179.132/JudgeOnline/problem.php?id=2756 题意:在一个 N*M 的棋盘上玩,每个格子有一个数.每次 选择两个相邻的格子,并使 ...

  3. 【BZOJ】【2756】【SCOI2012】奇怪的游戏

    网络流-最大流 这题……建模部分先略过 这道题是会卡时限的T_T俺的Dinic被卡了,在此放几篇很棒的讲网络流算法的文章,至于大家耳熟能详的论文就不放了…… http://www.cppblog.co ...

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

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

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

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

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

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

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

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

  8. bzoj 2756: [SCOI2012]奇怪的游戏

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

  9. BZOJ 2756 【SCOI2012】 奇怪的游戏

    题目链接:奇怪的游戏 一开始这道题想岔了……想到黑白染色后对总格子数按奇偶性分类讨论,然后没发现奇数个格子的可以直接解方程…… 首先可以发现每次操作是给相邻的两个格子权值加一,因此我们把棋盘黑白染色后 ...

随机推荐

  1. 【BZOJ3038】上帝造题的七分钟2 线段树

    根据一个数六次√必死,我们可以打标记死了就不管他了,于是有贡献的操作复杂度为O(n*logn*6),然而我们还有由于盲目修改造成的多余代价我们把每次查询的区间分成三部分前全死,中残,后全死,对于中残, ...

  2. BZOJ2631 tree(伍一鸣) LCT 秘制标记

    这个题一看就是裸地LCT嘛,但是我wa了好几遍,这秘制标记...... 注意事项:I.*对+有贡献 II.先下传*再下传+(因为我们已经维护了+,不能再让*对+产生贡献)III.维护+用到size # ...

  3. JQuery用鼠标选文字来发新浪微博

    最近注意到新浪博客有个小功能,就是当鼠标选中一段文字时会浮现一个小图片,点击这个图片可以把选中内容发送到新浪微博,一时兴起昨晚就写了一个Demo玩了一下,代码超简单,没优化,有兴趣的朋友可以自己改进. ...

  4. 非阻塞式JavaScript脚本及延伸知识

    JavaScript 倾向于阻塞浏览器某些处理过程,如HTTP 请求和界面刷新,这是开发者面临的最显著的性能问题.保持JavaScript文件短小,并限制HTTP请求的数量,只是创建反应迅速的网页应用 ...

  5. Django请求原理

    总结一下: 1. 进来的请求转入/hello/. 2. Django通过在ROOT_URLCONF配置来决定根URLconf. 3. Django在URLconf中的所有URL模式中,查找第一个匹配/ ...

  6. bzoj 5093 [Lydsy1711月赛]图的价值 NTT+第二类斯特林数

    [Lydsy1711月赛]图的价值 Time Limit: 30 Sec  Memory Limit: 256 MBSubmit: 245  Solved: 128[Submit][Status][D ...

  7. ES6学习笔记(五)—— 编程风格

    1. 块级作用域 let 取代 var —— let 只在声明的代码块内有效,而且不存在变量提升的效用 const 取代 let —— const 比较符合函数式编程的思想,运算不改变值,只是新建值: ...

  8. [洛谷P2127] 序列排序

    洛谷题目链接:序列排序 题目描述 小C有一个N个数的整数序列,这个序列的中的数两两不同.小C每次可以交换序列中的任意两个数,代价为这两个数之和.小C希望将整个序列升序排序,问小C需要的最小代价是多少? ...

  9. 【POJ 1719】 Shooting Contest (二分图匹配)

    题目链接 把每一列能射的两行和这一列连边,然后跑一边匈牙利就行了. #include <cstdio> #include <cstring> #include <algo ...

  10. 【洛谷 P2515】 [HAOI2010]软件安装 (缩点+树形背包)

    题目链接 看到代价和价值这两个关键词,肯定是首先要想到背包的. 但是图中并没有说这是棵树,所以先要\(Tarjan\)缩点,然后就是选课了,跑一遍树形背包就好了. 注意:缩点后应该是一个森林,应该用一 ...