题意:给定一个网格,一些格子是障碍不用管,剩余的格子是城市,你可以修建铁路,铁路的形状可以是直的或者弯的,也就是说可以以这个点为节点连接它四联通的其中两个方块。要求用一个或多个环来覆盖所有城市。对于有些关键点,如果这里是直轨道,会付出1的代价,如果不能覆盖,输出-1,否则输出最小代价。

这次该最小费用流了。x先考虑怎么判断有无解。很明显,既然是棋盘,想不染色不二分图都难。染成黑白后,对于黑点,S向其连2,黑点向周围的白点连1,白点向T连2,判断是否满流就好了。那么怎么计算代价呢?我们发现,如果要付出代价,那么一定是两个开口都给了同一列或者同一行,为了对此限制,我们拆点,分别管辖行和列。

如果这个点是黑关键点,我们向行对应的分身连一个容量1,费用0的边,再连一个容量1,费用1的边,表示如果只用一条边,不会产生费用,否则产生两条边的费用。列的话同理。

白关键点就不赘述了,其实也就是相较于反了一下。对于普通点,直接相对应的两个点分别连(2,0)就好了(因为没有限制),然后黑连向对应的白就好了。

跑费用流,不满流无解,满流输出费用即可。

 #include<bits/stdc++.h>
using namespace std;
#define INF 1e9
#define N 5005
#define id(i,j) ((i-1)*m+j)
int n,m,TOT,S,T,P,head[N],d[N],a[N],p[N],cnt;
bool vis[N];
queue<int>q;
char field[N][N];
inline int read(){
int x=,f=; char a=getchar();
while(a<'' || a>'') {if(a=='-') f=-; a=getchar();}
while(a>='' && a<='') x=x*+a-'',a=getchar();
return x*f;
}
struct edges{
int fr,to,cap,flow,cost,next;
}e[*N]; inline void insert(int u,int v,int f,int c){
e[cnt]=(edges){u,v,f,,c,head[u]};head[u]=cnt++;
e[cnt]=(edges){v,u,,,-c,head[v]};head[v]=cnt++;
}
inline bool spfa(){
memset(d,0x3f,sizeof(d));
d[S]=; a[S]=INF; q.push(S);
while(!q.empty()){
int x=q.front(); q.pop(); vis[x]=;
for(int i=head[x];i>=;i=e[i].next)
if(d[e[i].to]>d[x]+e[i].cost && e[i].flow<e[i].cap){
d[e[i].to]=d[x]+e[i].cost; p[e[i].to]=i;
a[e[i].to]=min(a[x],e[i].cap-e[i].flow);
if(!vis[e[i].to]) vis[e[i].to]=,q.push(e[i].to);
}
}
return d[T]<INF;
}
inline int mincf(){
int u=T;
while(u!=S){
e[p[u]].flow+=a[T];
e[p[u]^].flow-=a[T];
u=e[p[u]].fr;
}
TOT-=a[T];
return a[T]*d[T];
} int main(){
n=read(); m=read();
for(int i=;i<=n;i++)
scanf("%s",field[i]+);
S=; T=*n*m+; P=n*m; TOT=;
memset(head,-,sizeof(head));
for(int i=;i<=n;i++)
for(int j=;j<=m;j++){
if(field[i][j]=='w') continue;
if((i+j)%==){
insert(S,id(i,j),,); TOT+=;
if(field[i][j]!='C') insert(id(i,j),id(i,j)+*P,,),insert(id(i,j),id(i,j)+P,,);
else {
insert(id(i,j),id(i,j)+P,,); insert(id(i,j),id(i,j)+P,,);
insert(id(i,j),id(i,j)+*P,,); insert(id(i,j),id(i,j)+*P,,);
}
if(j-> && field[i][j-]!='w') insert(id(i,j)+P,id(i,j-)+P,,);
if(j+<=m && field[i][j+]!='w' ) insert(id(i,j)+P,id(i,j+)+P,,);
if(i-> && field[i-][j]!='w') insert(id(i,j)+*P,id(i-,j)+*P,,);
if(i+<=n && field[i+][j]!='w') insert(id(i,j)+*P,id(i+,j)+*P,,);
}else{
insert(id(i,j),T,,);
if(field[i][j]!='C') insert(id(i,j)+*P,id(i,j),,),insert(id(i,j)+P,id(i,j),,);
else{
insert(id(i,j)+P,id(i,j),,); insert(id(i,j)+P,id(i,j),,);
insert(id(i,j)+*P,id(i,j),,); insert(id(i,j)+*P,id(i,j),,);
}
}
}
int ans=;
while(spfa()) ans+=mincf();
if(TOT) ans=-;
printf("%d\n",ans);
}

最新文章

  1. Composer概述及其自动加载探秘
  2. PHP 原型模式
  3. ANDROID开发之问题积累及解决方案(一)
  4. 标签q
  5. TCP/IP协议(一)
  6. rails获取json内容
  7. PL/SQL developer 使用技巧汇总
  8. 用frame实现最基本的上中下三层布局,中间又分左右两部分.
  9. linux scp传文件
  10. CSS兼容的一些问题
  11. border-radius IE8兼容处理
  12. Linux 性能监测:IO
  13. 用汇编语言角度来理解C语言的一些问题
  14. Hbase--知识点总结3
  15. Java Number &amp; Math 类
  16. 隐藏字符 BOM
  17. Java基础96 ajax技术的使用
  18. mysql操作说明,插入时外键约束,快速删除
  19. linux信号处理相关知识
  20. C# 判断是否是在设计模式下有效的方法

热门文章

  1. centos6安装svn1.9
  2. GOLANG 基本数据类型 整型
  3. div居中方法
  4. web前端性能优化指南(转)
  5. (原创)Xilinx的ISE生成模块ngc网表文件
  6. Java多线程总结之线程安全队列Queue
  7. NSScanner知悉 (转)
  8. Ajax form表单提交
  9. 转:logBack.xml配置路径
  10. iOS开发 二维码生成