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 第一行 ...
随机推荐
- CentOS配置vsftpd遇到550错误的解决办法
默认情况下用root是无法连接的,你可以创建一个非root帐户登录,但是登录是可以登录,却没有办法创建或是上传文件.有人说,可以把 SELinux关掉,可是这样未免也有点尺度太大了,其实是SELinu ...
- Python-Day1 Python基础学习
一.Python3.5.X安装 1.Windows Windows上找度娘搜索“Python for windows下载”就OK了,安装的时候可以勾选设置环境变量,也可以安装完手动设置,这样在cmd中 ...
- Android类库常用类库一览
在Android SDK中包括很多包文件,通过了解这些包的功能也有助于了解可以开发的功能. 在Android类库中,各种包写成android.*的方式,重要包的描述如下所示: android.app ...
- 2016 Multi-University Training Contest 1 GCD RMQ+二分(预处理)
链接:http://acm.hdu.edu.cn/showproblem.php?pid=5726 题意:有N(N <= 100,000),之后有Q(Q <= 100,000)个区间查询[ ...
- WebApp之 apple-touch-icon
在iPhone,iPad,iTouch的safari上可以使用添加到主屏按钮将网站添加到主屏幕上.apple-touch-icon是IOS设备的私有标签,如果设置了相应apple-touch-icon ...
- OFBiz进阶之HelloWorld(五)创建新实体
参考文档 https://cwiki.apache.org/confluence/display/OFBIZ/OFBiz+Tutorial+-+A+Beginners+Development+Guid ...
- 【bzoj3110】[Zjoi2013]K大数查询
Description 有N个位置,M个操作.操作有两种,每次操作如果是1 a b c的形式表示在第a个位置到第b个位置,每个位置加入一个数c如果是2 a b c形式,表示询问从第a个位置到第b个位置 ...
- swift基础--数组、字典
(1)初始化 (2)新增.修改.删除 (3)清空 (4)遍历 var array1 = ["x","y","z"] var array2:[ ...
- 手写快速排序(QuickSort)
#include<iostream> #include<stdio.h> #include<algorithm> using namespace std; int ...
- ByteBuffer的allocate和allocateDirect区别
ByteBuffer的allocate和allocateDirect区别 在Java中当我们要对数据进行更底层的操作时,通常是操作数据的字节(byte)形式,这时常常会用到ByteBuffer这样一个 ...