2668: [cqoi2012]交换棋子

Time Limit: 3 Sec  Memory Limit: 128 MB
Submit: 1112  Solved: 409
[Submit][Status][Discuss]

Description

有一个nm列的黑白棋盘,你每次可以交换两个相邻格子(相邻是指有公共边或公共顶点)中的棋子,最终达到目标状态。要求第i行第j列的格子只能参与mi,j次交换。

Input

第一行包含两个整数nm(1<=nm<=20)。以下n行为初始状态,每行为一个包含m个字符的01串,其中0表示黑色棋子,1表示白色棋子。以下n行为目标状态,格式同初始状态。以下n行每行为一个包含m个0~9数字的字符串,表示每个格子参与交换的次数上限。
 

Output

输出仅一行,为最小交换总次数。如果无解,输出-1。

Sample Input

3 3
110
000
001
000
110
100
222
222
222

Sample Output

4

HINT

 

Source

 

[Submit][Status][Discuss]

对于每一个格子,拆成三个点,分别为x,y,z。

其中,x为格子间转移边的入点,z为出点,y为与S和T的连接点。

S向每个原图中黑色,新图中白色的y点连一条容量为1,费用为0的边,流过这条边表示把该棋子换走替代其他棋子。

T向每个原图中白色,新图中黑色的y点连一条容量为1,费用为0的边,流经这条边表示用其他棋子把该棋子换走。

所有原图中黑色,新图中白色的点,x向y连容量为$\frac{limit_{i,j}}{2}$,费用为0的边,表示至多接受这么多替换;y向z连容量为$\frac{limit_{i,j}+1}{2}$,费用为0的边,表示之多提供这么多替换。所有原图中黑色,新图中白色的点,和这个反着连。新图原图中一样的,容量则都是$\frac{limit_{i,j}}{2}$。

按照八联通,所有相邻点对,连接其对应为x,z即可,容量无限,费用为1,表示交换一次的代价为1。

 #include <bits/stdc++.h>
#define rep(a,b,c) for(register int a=b;a<=c;++a)
const int mxn = , siz = , inf = 1e9, mv[][] = {{,},{,},{,},{,-}};
char a[mxn][mxn], b[mxn][mxn], c[mxn][mxn];
int n, m, d[mxn][mxn], id[mxn][mxn][], ID, cnt0, cnt1;
int s, t, tot, hd[siz], to[siz], nt[siz], fl[siz], vl[siz], dis[siz], pre[siz], que[siz], inq[siz], cost;
inline bool legal(int x, int y) { return x >= && x <= n && y >= && y <= m; }
inline void add(int u, int v, int f, int w) {
nt[tot] = hd[u]; to[tot] = v; fl[tot] = f; vl[tot] = +w; hd[u] = tot++;
nt[tot] = hd[v]; to[tot] = u; fl[tot] = ; vl[tot] = -w; hd[v] = tot++;
}
inline bool spfa(void) {
register int head = , tail = ;
rep(i, s, t)dis[i] = inf, inq[i] = ;
dis[que[tail++] = s] = , inq[s] = , pre[s] = -;
while (head != tail) {
int u = que[head++], v, i; inq[u] = ;
for (i = hd[u]; ~i; i = nt[i])if (fl[i] && dis[v = to[i]] > dis[u] + vl[i]) {
dis[v] = dis[u] + vl[i], pre[v] = i^; if (!inq[v])inq[que[tail++] = v] = ;
}
}
return dis[t] < inf;
}
inline int maxFlow(void) {
int ret = ;
while (spfa()) {
int flow = inf, i;
for (i = pre[t]; ~i; i = pre[to[i]])if (flow > fl[i^])flow = fl[i^];
for (i = pre[t]; ~i; i = pre[to[i]])fl[i] += flow, fl[i^] -= flow;
ret += flow, cost += flow * dis[t];
}
return ret;
}
signed main(void) {
scanf("%d%d", &n, &m);
s = , t = n * m * + ;
memset(hd, -, sizeof(hd));
rep(i, , n)scanf("%s", a[i] + );
rep(i, , n)scanf("%s", b[i] + );
rep(i, , n)scanf("%s", c[i] + );
rep(i, , n)rep(j, , m)d[i][j] = c[i][j] - '';
rep(i, , n)rep(j, , m)rep(k, , )id[i][j][k] = ++ID;
rep(i, , n)rep(j, , m) {
if (a[i][j] == '' && b[i][j] == '')
add(s, id[i][j][], , ), ++cnt0,
add(id[i][j][], id[i][j][], d[i][j] >> , ),
add(id[i][j][], id[i][j][], (d[i][j] + ) >> , );
if (a[i][j] == '' && b[i][j] == '')
add(id[i][j][], t, , ), ++cnt1,
add(id[i][j][], id[i][j][], (d[i][j] + ) >> , ),
add(id[i][j][], id[i][j][], d[i][j] >> , );
if (a[i][j] == b[i][j])
add(id[i][j][], id[i][j][], d[i][j] >> , ),
add(id[i][j][], id[i][j][], d[i][j] >> , );
}
rep(i, , n)rep(j, , m)rep(k, , )if (legal(i + mv[k][], j + mv[k][]))
add(id[i][j][], id[i + mv[k][]][j + mv[k][]][], inf, ),
add(id[i + mv[k][]][j + mv[k][]][], id[i][j][], inf, );
printf("%d\n", cnt0 == cnt1 && cnt0 == maxFlow() ? cost : -);
}

@Author: YouSiki

BZOJ 2668: [cqoi2012]交换棋子的更多相关文章

  1. BZOJ 2668 [cqoi2012]交换棋子 | 最小费用最大流

    传送门 BZOJ 2668 题解 同时分别限制流入和流出次数,所以把一个点拆成三个:入点in(x).中间点mi(x).出点ou(x). 如果一个格子x在初始状态是黑点,则连(S, mi(x), 1, ...

  2. BZOJ.2668.[CQOI2012]交换棋子(费用流zkw)

    题目链接 首先黑白棋子的交换等价于黑棋子在白格子图上移动,都到达指定位置. 在这假设我们知道这题用网络流做. 那么黑棋到指定位置就是一条路径,考虑怎么用流模拟出这条路径. 我们发现除了路径的起点和终点 ...

  3. 2668: [cqoi2012]交换棋子

    Description 有一个n行m列的黑白棋盘,你每次可以交换两个相邻格子(相邻是指有公共边或公共顶点)中的棋子,最终达到目标状态.要求第i行第j列的格子只能参与mi,j次交换. Input 第一行 ...

  4. BZOJ2668: [cqoi2012]交换棋子

    题解: 可以戳这里:http://www.cnblogs.com/zig-zag/archive/2013/04/21/3033485.html 其实自己yy一下就知道这样建图的正确性了. 感觉太神奇 ...

  5. [cqoi2012]交换棋子

      2668: [cqoi2012]交换棋子 Time Limit: 3 Sec  Memory Limit: 128 MBSubmit: 1334  Solved: 518[Submit][Stat ...

  6. 【BZOJ2668】[cqoi2012]交换棋子 费用流

    [BZOJ2668][cqoi2012]交换棋子 Description 有一个n行m列的黑白棋盘,你每次可以交换两个相邻格子(相邻是指有公共边或公共顶点)中的棋子,最终达到目标状态.要求第i行第j列 ...

  7. 洛谷 P3159(BZOJ 2668)[CQOI2012]交换棋子

    有一个\(n\)行\(m\)列的黑白棋盘,你每次可以交换两个相邻格子(相邻是指有公共边或公共顶点)中的棋子,最终达到目标状态.要求第\(i\)行第\(j\)列的格子只能参与\(m[i][j]\)次交换 ...

  8. BZOJ2668:[CQOI2012]交换棋子——题解

    http://www.lydsy.com/JudgeOnline/problem.php?id=2668 https://www.luogu.org/problemnew/show/P3159#sub ...

  9. P3159 [CQOI2012]交换棋子

    思路 相当神奇的费用流拆点模型 最开始我想到把交换黑色棋子看成一个流流动的过程,流从一个节点流向另一个节点就是交换两个节点,然后把一个位置拆成两个点限制流量,然后就有了这样的建图方法 S向所有初始是黑 ...

随机推荐

  1. POJ2251-Dungeon Master(3维BFS)

    You are trapped in a 3D dungeon and need to find the quickest way out! The dungeon is composed of un ...

  2. maven项目的标准目录结构

    maven项目的标准目录结构如下:

  3. JavaScript学习要点

    Javascript相关内容 1.序列化--json - stringify() 将对象转换为字符串 - parse() 将字符串转换为对象 list=[11,22,33,44,55]; 结果:(5) ...

  4. 2019第十届蓝桥杯C++B组题解(赛后重写的,不确保答案正确性,仅供参考)

    先说一下这次的感受吧,我们考场比较乱,开始比赛了,还有的电脑有故障,(向这些人发出同情),第一次认真参加比赛,真正比赛的时候感觉没有那么正式,很乱,各种小问题,(例如博主就没找到题目在哪里,找到后又不 ...

  5. python 拾遗

    三引号 可以当做多行字符串使用.在类或者方法中用的时候,可以当做docstrings来使用.具体的规则为: 该文档字符串所约定的是一串多行字符串,其中第一行以某一大写字母开始,以句号结束.第二行为空行 ...

  6. java浮点数存储

    转自: [解惑]剖析float型的内存存储和精度丢失问题 1.小数的二进制表示问题 首先我们要搞清楚下面两个问题: (1)  十进制整数如何转化为二进制数 算法很简单.举个例子,11表示成二进制数: ...

  7. 【quickhybrid】Android端的项目实现

    前言 前文中就有提到,Hybrid模式的核心就是在原生,而本文就以此项目的Android部分为例介绍Android部分的实现. 提示,由于各种各样的原因,本项目中的Android容器确保核心交互以及部 ...

  8. sprint3(第一天)

    1.今天计划了sprint3要做的内容: 整合前台和后台,然后发布让用户使用,然后给我们反馈再进行改进 2.backlog表格: ID Name Est How to demo 1 实现用户登录与权限 ...

  9. 忘记本地MySQL数据库密码的解决方案。

    忘记本地MySQL数据库密码,解决方案,分以下10个步骤: 参考链接:                  https://blog.csdn.net/weidong_y/article/details ...

  10. 2018-2019-20172321 《Java软件结构与数据结构》第九周学习总结

    2018-2019-20172321 <Java软件结构与数据结构>第九周学习总结 教材学习内容总结 第15章 图 无向图 图由顶点和边组成. 顶点由名字或标号来表示,如:A.B.C.D: ...