博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
FZU 2143 Board Game
阅读量:6059 次
发布时间:2019-06-20

本文共 2468 字,大约阅读时间需要 8 分钟。

这题好难。。第一次遇到这样的建图,表示是看了好多题解才懂的。

首先这是一个费用流的题,但是请注意,并不是达到最大流时候的最小费用

首先分析每一个位置对答案做出的贡献,即a[i][j]*a[i][j]-2*a[i][j]*b[i][j]+b[i][j]*b[i][j],最后一项是个常数,所以我们只要计算前两项的总和的最小值。

a[i][j]表示的意义是对i,j这个位置的操作次数

假设操作了p次,这个点对答案做出的贡献是p*p-2*p*b[i][j]+b[i][j]*b[i][j]

假设操作了p-1次,这个点对答案做出的贡献是(p-1)*(p-1)-2*(p-1)*b[i][j]+b[i][j]*b[i][j]

第p次的贡献与第p-1次的贡献的差值是2*p-1-2*b[i][j]。

建图:

首先对矩阵间隔染上黑白两色,我们只对黑色格子进行操作,与黑色格子相邻的白色格子可以连动。

原点到每一个黑色格子连k条边,容量为1,第p条的费用是2*p-1-2*b[i][j],表示进行了第p次操作,在进行了p-1次操作的基础上,对答案的贡献新增加了2*p-1-2*b[i][j]

每一个白格子到汇点连k条边,容量为1,第p条的费用是2*p-1-2*b[i][j],意义同上。

请注意:2*p-1-2*b[i][j]这个式子是随着p增大而增大的,所以流量必然是从费用小的开始流,恰好与题目操作吻合。

黑色格子与他相邻的白格子连变,容量为INF,费用为0,这样能保证一个黑格子有流量流过的时候必然会把流量流向一个相邻白格子,与题目操作吻合。

接下里就是做一次费用流,但是请注意:

费用流算法结束的标志并不是SPFA的时候不能达到汇点!

而是当达到汇点的距离>=0的时候就结束了,这个时候答案已经最小了。

再继续流下去,答案会变大而不是减小。

#include
#include
#include
#include
#include
#include
using namespace std;const int maxn=200;int n,m,k;int b[15][15];int id[15][15];const int INF=0x7FFFFFFF;struct Edge{ int from,to,cap,flow,cost;};int len,s,t;vector
edges;vector
G[maxn];int inq[maxn];int d[maxn];int p[maxn];int a[maxn];int use[maxn];int ans;int dir[4][2]={ { 1,0}, { -1,0}, { 0,-1}, { 0,1},};void init(){ for(int i=0; i
Q; Q.push(s); while(!Q.empty()) { int u=Q.front(); Q.pop(); inq[u]=0; for(int i=0; i
e.flow&&d[e.to]>d[u]+e.cost) { d[e.to]=d[u]+e.cost; p[e.to]=G[u][i]; a[e.to]=min(a[u],e.cap-e.flow); if(!inq[e.to]) { Q.push(e.to); inq[e.to]=1; } } } } if(d[t]>=0) return false; flow+=a[t]; cost+=d[t]*a[t]; int u=t; while(u!=s) { edges[p[u]].flow+=a[t]; edges[p[u]^1].flow-=a[t]; u=edges[p[u]].from; } return true;}void Mincost (int s,int t){ int flow=0,cost=0; while(BellmanFord(s,t,flow,cost)); ans=ans+cost;}bool P(int a,int b){ if(a>=1&&a<=n&&b>=1&&b<=m) return 1; return 0;}int main(){ int T; scanf("%d",&T); for(int cas=1; cas<=T; cas++) { init(); scanf("%d%d%d",&n,&m,&k); ans=0; int ID=1; for(int i=1; i<=n; i++) for(int j=1; j<=m; j++) { scanf("%d",&b[i][j]); ans=ans+b[i][j]*b[i][j]; id[i][j]=ID++; } s=0; t=n*m+1; for(int i=1; i<=n; i++) { for(int j=1; j<=m; j++) { if((i+j)%2==1) { for(int p=1; p<=k; p++) Addedge(s,id[i][j],1,2*p-1-2*b[i][j]); for(int d=0; d<4; d++) { int x=i+dir[d][0],y=j+dir[d][1]; if(P(x,y)) Addedge(id[i][j],id[x][y],INF,0); } } else { for(int p=1; p<=k; p++) Addedge(id[i][j],t,1,2*p-1-2*b[i][j]); } } } Mincost(s,t); printf("Case %d: %d\n",cas,ans); } return 0;}

 

转载于:https://www.cnblogs.com/zufezzt/p/5393392.html

你可能感兴趣的文章
让IE6 IE7 IE8 IE9 IE10 IE11支持Bootstrap的解决方法
查看>>
Just a complaint about the college examinations
查看>>
Oracle PLSQL Demo - 10.For Loop遍历游标[FOR LOOP CURSOR]
查看>>
U_boot 的 bootcmd 和bootargs参数详解
查看>>
android 删除SD卡或手机的缓存图像和文件夹
查看>>
SharePoint 2013 配置基于表单的身份认证
查看>>
Git 忽略某个目录中的文件,同时保留这个目录
查看>>
Android它Service
查看>>
DrawItem
查看>>
php之数据类型自动转换
查看>>
***PHP请求服务curl以及json的解析
查看>>
Node.js爬虫数据抓取 -- 问题总结
查看>>
.NET 4.6的RyuJIT尾递归优化的Bug
查看>>
如何配置多个ssh key
查看>>
Android Sqlite数据库执行插入查询更新删除的操作对比
查看>>
【iOS】iOS的iTunes文件共享,在程序Document路径
查看>>
数据仓库备份思路
查看>>
URAL 1780 G - Gray Code 找规律
查看>>
Programming Internal Flash Over the Serial Wire Debug <SWD> Interface -- EFM32
查看>>
前端乱炖,总结一些细节点
查看>>