BZOJ2668: [cqoi2012]交换棋子
题解:
可以戳这里:http://www.cnblogs.com/zig-zag/archive/2013/04/21/3033485.html
其实自己yy一下就知道这样建图的正确性了。
感觉太神奇,居然还能拆成3个点
orzzzzzzzzzzzzzzzzzzzzzzzzz
跪跪跪跪跪跪跪跪
代码:
#include<cstdio> #include<cstdlib> #include<cmath> #include<cstring> #include<algorithm> #include<iostream> #include<vector> #include<map> #include<set> #include<queue> #include<string> #define inf 1000000000 #define maxn 200000+5 #define maxm 200000+5 #define eps 1e-10 #define ll long long #define pa pair<int,int> #define for0(i,n) for(int i=0;i<=(n);i++) #define for1(i,n) for(int i=1;i<=(n);i++) #define for2(i,x,y) for(int i=(x);i<=(y);i++) #define for3(i,x,y) for(int i=(x);i>=(y);i--) #define for4(i,x) for(int i=head[x],y;i;i=e[i].next) #define mod 1000000007 using namespace std; inline int read() { int x=,f=;char ch=getchar(); while(ch<''||ch>''){if(ch=='-')f=-;ch=getchar();} while(ch>=''&&ch<=''){x=*x+ch-'';ch=getchar();} return x*f; }
int n,m,k,mincost,tot,s,t,head[maxn],d[maxn],from[*maxm];
bool v[maxn];
queue<int>q;
int a[][][],num[][][],cnt[];
struct edge{int from,go,next,v,c;}e[*maxm];
void add(int x,int y,int v,int c)
{
e[++tot]=(edge){x,y,head[x],v,c};head[x]=tot;
e[++tot]=(edge){y,x,head[y],,-c};head[y]=tot;
}
bool spfa()
{
for (int i=s;i<=t;i++){v[i]=;d[i]=inf;}
q.push(s);d[s]=;v[s]=;
while(!q.empty())
{
int x=q.front();q.pop();v[x]=;
for (int i=head[x],y;i;i=e[i].next)
if(e[i].v&&d[x]+e[i].c<d[y=e[i].go])
{
d[y]=d[x]+e[i].c;from[y]=i;
if(!v[y]){v[y]=;q.push(y);}
}
}
return d[t]!=inf;
}
void mcf()
{
mincost=;
while(spfa())
{
int tmp=inf;
for(int i=from[t];i;i=from[e[i].from]) tmp=min(tmp,e[i].v);
mincost+=d[t]*tmp;
for(int i=from[t];i;i=from[e[i].from]){e[i].v-=tmp;e[i^].v+=tmp;}
}
}
const int dx[]={,,,-,,,-,-};
const int dy[]={,-,,,-,,,-}; int main() {
n=read();m=read();s=;t=*n*m+;
for0(k,)for1(i,n)for1(j,m)
{
char ch=getchar();
while(ch<''||ch>'')ch=getchar();
a[i][j][k]=ch-'';
//cout<<i<<' '<<j<<' '<<k<<' '<<a[i][j][k]<<endl;
num[i][j][k]=++tot;
cnt[k]+=a[i][j][k];
}
//cout<<tot<<' '<<s<<' '<<t<<endl;
tot=;
for1(i,n)for1(j,m)
{
if(a[i][j][]&&a[i][j][])a[i][j][]=a[i][j][]=;
if(a[i][j][])
{
add(num[i][j][],num[i][j][],a[i][j][]/,);
add(num[i][j][],num[i][j][],(a[i][j][]+)/,);
add(s,num[i][j][],,);
}
else if(a[i][j][])
{
add(num[i][j][],num[i][j][],(a[i][j][]+)/,);
add(num[i][j][],num[i][j][],a[i][j][]/,);
add(num[i][j][],t,,);
}
else
{
add(num[i][j][],num[i][j][],a[i][j][]/,);
add(num[i][j][],num[i][j][],a[i][j][]/,);
}
for0(k,)
{
int x=i+dx[k],y=j+dy[k];
if(x<||x>n||y<||y>m)continue;
add(num[i][j][],num[x][y][],inf,);
}
}
if(cnt[]!=cnt[]){printf("-1\n");return ;}
mcf();
printf("%d\n",mincost); return ; }
2668: [cqoi2012]交换棋子
Time Limit: 3 Sec Memory Limit: 128 MB
Submit: 673 Solved: 235
[Submit][Status]
Description
Input
Output
Sample Input
110
000
001
000
110
100
222
222
222
Sample Output
BZOJ2668: [cqoi2012]交换棋子的更多相关文章
- BZOJ2668 [cqoi2012]交换棋子 【费用流】
题目链接 BZOJ2668 题解 容易想到由\(S\)向初始的黑点连边,由终态的黑点向\(T\)连边,然后相邻的点间连边 但是这样满足不了交换次数的限制,也无法计算答案 考虑如何满足一个点的交换次数限 ...
- BZOJ2668:[CQOI2012]交换棋子(费用流)
题目描述 有一个n行m列的黑白棋盘,你每次可以交换两个相邻格子(相邻是指有公共边或公共顶点)中的棋子,最终达到目标状态.要求第i行第j列的格子只能参与mi,j次交换. 输入输出格式 输入格式: 第一行 ...
- BZOJ2668: [cqoi2012]交换棋子(费用流)
Description 有一个n行m列的黑白棋盘,你每次可以交换两个相邻格子(相邻是指有公共边或公共顶点)中的棋子,最终达到目标状态.要求第i行第j列的格子只能参与mi,j次交换. Input 第一行 ...
- 【BZOJ2668】[cqoi2012]交换棋子 费用流
[BZOJ2668][cqoi2012]交换棋子 Description 有一个n行m列的黑白棋盘,你每次可以交换两个相邻格子(相邻是指有公共边或公共顶点)中的棋子,最终达到目标状态.要求第i行第j列 ...
- 【BZOJ-2668】交换棋子 最小费用最大流
2668: [cqoi2012]交换棋子 Time Limit: 3 Sec Memory Limit: 128 MBSubmit: 1055 Solved: 388[Submit][Status ...
- BZOJ 2668: [cqoi2012]交换棋子
2668: [cqoi2012]交换棋子 Time Limit: 3 Sec Memory Limit: 128 MBSubmit: 1112 Solved: 409[Submit][Status ...
- [cqoi2012]交换棋子
2668: [cqoi2012]交换棋子 Time Limit: 3 Sec Memory Limit: 128 MBSubmit: 1334 Solved: 518[Submit][Stat ...
- BZOJ2668:[CQOI2012]交换棋子——题解
http://www.lydsy.com/JudgeOnline/problem.php?id=2668 https://www.luogu.org/problemnew/show/P3159#sub ...
- [bzoj2668] [洛谷P3159] [cqoi2012] 交换棋子
Description 有一个n行m列的黑白棋盘,你每次可以交换两个相邻格子(相邻是指有公共边或公共顶点)中的棋子,最终达到目标状态.要求第i行第j列的格子只能参与mi,j次交换. Input 第一行 ...
随机推荐
- 《C和指针》 读书笔记 -- 第10章 结构和联合
1.聚合数据类型能够同时存储超过一个的单独数据,c提供了两种类型的聚合数据类型,数组和结构. 2.[1] struct SIMPLE { int a; }; struct SIMPLE x; [2] ...
- Linux 查看某个用户的进程
Linux 查看某个用户的进程 To view only the processes owned by a specific user, use the following command: top ...
- 请给出一个左侧定宽右侧自适应的HTML结构及样式
<!DOCTYPE HTML><html><head><meta http-equiv="Content-Type" content=&q ...
- nodejs phantom add click event
page.evaluate( function() { // find element to send click to var element = document.querySelector( ' ...
- springMVC数据封装成POJO
springMVC把前台的数据封装为POJO与struts2的封装形式不同.struts2需要在控制器声明需封装的POJO,而springMVC不需要任何准备工作,只需在相应的方法的参数中加上需封装的 ...
- IOS xib生成界面和代码生成界面两种方式混合
应用程序代理类 WKAppDelegate.m // // WKAppDelegate.m // HelloWorld // // Created by easy5 on 13-9-18. // Co ...
- WPF SplitButton 的杂七杂八
原文: http://www.codeproject.com/Articles/20612/A-WPF-SplitButton SplitButton.cs using System; using S ...
- uva 10608
简单并查集 水水..... #include <cstdio> #include <cstring> #define maxn 30005 int fa[maxn],ans[ ...
- Linux---Ls命令 初级实现
By xxx0624Done: ls ls -a ls -l ls /tmp ls -R ls -t FileName color FileName o ...
- linux mysql为root用户初始化密码和改变root密码
初始化密码: 由于安装MySQL完后,MySQL会自动提供一个不带密码的root用户,为了安全起见给root设置密码: #mysqladmin -u root password 123 (123为密码 ...