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. int str input的运用

  2. .Net 如何访问主流的各大数据库

    做过开发的都知道,.NET基本可以理解是和MSSQL,windows服务器属于一个好的搭档,正如PHP和MYSQL,LIUNX等也可以理解是一个完美搭配:但是在实际的开发中并不完全是这样的,如果你是学 ...

  3. HTML基础范例

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  4. 买卖股票的最佳时机 II

    int maxProfit(int* prices, int pricesSize) { ; ; i < pricesSize - ; i++) { ]) { continue; } else ...

  5. 帝国cms后台集成ueditor编辑器

    我更换成百度编辑器的原因有以下几点:1.使用百度编辑器的图片粘贴上传功能,这个功能实在是太有必要了,有开发的过程中或上传的过程中,通常用qq直接截图,直接放到文章上面,避免了再放到本地保存的情况,真是 ...

  6. Spring学习(5):DI的配置

    一.  一些概念 应用程序中说的依赖一般指类之间的关系. 泛化:表示类与类之间的继承关系.接口与接口之间的继承关系: 实现:表示类对接口的实现: 依赖:当类与类之间有使用关系时就属于依赖关系,不同于关 ...

  7. CHAPTER 24 History of Our Planet 第24章 我们行星的历史

    CHAPTER 24 History of Our Planet 第24章 我们行星的历史 Uncovering the bones of ancient beasts is only part of ...

  8. init命令详解

    基础命令学习目录首页 1.手动输入命令会执行相关操作   #init 0 - 停机(千万不能把initdefault 设置为0 )   #init 1 - 单用户模式   #init 2 - 多用户, ...

  9. 王者荣耀交流协会PSP Daily项目Postmortem结果

    王者荣耀交流协会PSP Daily项目Postmortem结果 整理:王超 设想和目标 1.       我们的软件要解决什么问题?是否定义得很清楚?是否对典型用户和典型场景有清晰的描述? PSP D ...

  10. [buaa-SE-2017]结对项目-数独程序扩展

    结对项目-数独程序扩展 step1~step3:github:SE-Sudoku-Pair-master step4:github:SE-Sudoku-Pair-dev-combine step5:g ...