4261: 建设游乐场

Time Limit: 50 Sec  Memory Limit: 256 MB
Submit: 38  Solved: 16
[Submit][Status][Discuss]

Description

现在有一大块土地,可以看成N*M的方格。在这块土地上,有些格子内是崎岖的山地,无法建造任何东西;其他格子都是平原。现在打算在这块土地上建设一个游乐园。游乐园由若干条闭合的过山车轨道组成,每个平原格子都要铺一截轨道,为下列 6 种类型中的一种:
(每张图表示一块平原格子,图内网格线为辅助线,无实际意义。)
 
其中前 2 种为直轨道,后 4 种为弯轨道。显然对游客来说,弯轨道更加刺激。
 
由于每块格子风景各不相同,经过一番研究,现给了N*M个方格中的每个格子一个评估值,意义为:如果该格子修建弯轨道,会给游客们带来多少的愉悦值。现需要一名设计师,帮他设计一种最优的轨道建设方案,使所有格子给游客们带来的愉悦值之和尽量大。(如果没有合法方案,输出 -1)

Input

第一行两个正整数 n, m。
接下来 n 行,每行 m 个数,描述了整块土地。其中 1 表示山地,0 表示平原。接下来 n 行,每行 m 个非负整数,第 i 行第 j 个为 Vi,j,表示格子 (i,j) 修建弯轨道能给游客们带来的愉悦值。

Output

一行一个数,表示最优设计方案中给游客们带来的愉悦值之和。

Sample Input

3 3
1 1 1
1 0 0
1 0 0
48 94 1
78 78 81
1 12 60

Sample Output

231

HINT

N<=150,M<=30,Vi,j<=100

Source

分析:

好题~~~

考虑构成若干个环的必要条件:每个格子的管道都要链接相邻的两个格子...然后把一个格子拆成两个管子,如果两个管子方向不同,则可以得到价值,如果相同就无法得到...

既然是网格图,那么我们考虑黑白染色,对于每个必须放的格子,我们把这个格子拆成三个点,一个点叫做控制点,一个代表竖向管道,一个代表横向管道...

然后对于每个黑点,从$S$向黑点的控制点连边$<S,id[i][j][0],2,0>$,代表连接两个格子,$<id[i][j][0],id[i][j][1],1,val>$,$<id[i][j][0],id[i][j][1],id[i][j][2],1,val>$,$<id[i][j][0],id[i][j][1],1,0>$,$<id[i][j][0],id[i][j][2],1,0>$...

对于每个白点,从白点的控制点向$T$连边$<id[i][j][0],T,2,0>$,然后$1$、$2$的连边和黑点相反...

然后对于每个黑点,我们把竖向点向上下的相邻格子个竖点连边$<1,0>$,横向点向左右的横向点连边$<1,0>$...

这样,如果满流并且黑白点相同的话就一定是合法解,因为代表每个格子都连接了相邻的两个格子...然后如果一个各自选择了相同方向的管道,那么会得到$val$的收益,如果选择不同的,会得到$2*val$的收益,所以求出最大费用之后减去$\sum val$就好了...

代码:

#include<algorithm>
#include<iostream>
#include<cstring>
#include<cstdio>
#include<queue>
//by NeighThorn
#define inf 0x3f3f3f3f
#define min(a,b) a>b?b:a
using namespace std; const int N=150+5,maxn=50000+5,maxm=1000000+5; int n,m,S,T,cnt,tat,tot,sum,mp[N][N],id[N][N][3],val[N][N];
int w[maxm],fl[maxm],hd[maxn],to[maxm],nxt[maxm],Min[maxn],dis[maxn],vis[maxn],from[maxn];
int mv[4][2]={1,0,-1,0,0,1,0,-1}; inline int read(void){
char ch=getchar();int x=0;
while(!(ch>='0'&&ch<='9')) ch=getchar();
while(ch>='0'&&ch<='9') x=x*10+ch-'0',ch=getchar();
return x;
} inline void add(int x,int y,int l,int s){
w[cnt]= s;fl[cnt]=l;to[cnt]=y;nxt[cnt]=hd[x];hd[x]=cnt++;
w[cnt]=-s;fl[cnt]=0;to[cnt]=x;nxt[cnt]=hd[y];hd[y]=cnt++;
} inline bool spfa(void){
memset(Min,inf,sizeof(Min));
memset(dis,inf,sizeof(dis));
queue<int> q;q.push(S);dis[S]=0;vis[S]=1;
while(!q.empty()){
int top=q.front();q.pop();vis[top]=0;
for(int i=hd[top];i!=-1;i=nxt[i])
if(dis[to[i]]>dis[top]+w[i]&&fl[i]){
from[to[i]]=i;
dis[to[i]]=dis[top]+w[i];
Min[to[i]]=min(Min[top],fl[i]);
if(!vis[to[i]])
vis[to[i]]=1,q.push(to[i]);
}
}
return dis[T]!=inf;
} inline int find(void){
for(int i=T;i!=S;i=to[from[i]^1])
fl[from[i]]-=Min[T],fl[from[i]^1]+=Min[T];
return Min[T]*dis[T];
} inline int mcmf(void){
int res=0,flow=0;
while(spfa())
res+=find(),flow+=Min[T];
if(flow!=tot) return -1;
return -res-sum;
} signed main(void){
n=read();m=read();
for(int i=1;i<=n;i++)
for(int j=1;j<=m;j++)
mp[i][j]=read();
for(int i=0;i<=n+1;i++) mp[i][0]=mp[i][m+1]=1;
for(int i=0;i<=m+1;i++) mp[0][i]=mp[n+1][i]=1;
for(int i=1;i<=n;i++)
for(int j=1;j<=m;j++){
val[i][j]=read();
if(!mp[i][j])
sum+=val[i][j];
}
for(int i=1;i<=n;i++)
for(int j=1;j<=m;j++)
if(!mp[i][j])
for(int k=0;k<3;k++)
id[i][j][k]=++tot;
S=0,T=++tot;tot=0;
memset(hd,-1,sizeof(hd));
for(int i=1;i<=n;i++)
for(int j=1;j<=m;j++)
if(!mp[i][j]){
if((i+j)&1)
tat+=2,
add(S,id[i][j][0],2,0),
add(id[i][j][0],id[i][j][1],1,0),
add(id[i][j][0],id[i][j][2],1,0),
add(id[i][j][0],id[i][j][1],1,-val[i][j]),
add(id[i][j][0],id[i][j][2],1,-val[i][j]);
else
tot+=2,
add(id[i][j][0],T,2,0),
add(id[i][j][1],id[i][j][0],1,0),
add(id[i][j][2],id[i][j][0],1,0),
add(id[i][j][1],id[i][j][0],1,-val[i][j]),
add(id[i][j][2],id[i][j][0],1,-val[i][j]);
}
for(int i=1;i<=n;i++)
for(int j=1;j<=m;j++)
if(!mp[i][j]&&((i+j)&1)){
for(int k=0,x,y;k<4;k++){
x=i+mv[k][0],y=j+mv[k][1];
if(!mp[x][y]){
if(k<2)
add(id[i][j][1],id[x][y][1],1,0);
else
add(id[i][j][2],id[x][y][2],1,0);
}
}
}
if(tat!=tot) return puts("-1"),0;
printf("%d\n",mcmf());
return 0;
}

  


By NeighThorn

BZOJ 4261: 建设游乐场的更多相关文章

  1. bzoj 4261: 建设游乐场 费用流

    题目 现在有一大块土地,可以看成N*M的方格.在这块土地上,有些格子内是崎岖的山地,无法建造任何东西:其他格子都是平原.现在打算在这块土地上建设一个游乐园.游乐园由若干条闭合的过山车轨道组成,每个平原 ...

  2. 【BZOJ-4261】建设游乐场 最大费用最大流

    4261: 建设游乐场 Time Limit: 50 Sec  Memory Limit: 256 MBSubmit: 21  Solved: 8[Submit][Status][Discuss] D ...

  3. BZOJ4261 : 建设游乐场

    将图黑白染色,每个点拆成两个点,分别表示水平和竖直方向,再增加一个点以控制流量,那么每个格子都需要找两个方向去连接. $S$到每个黑点的控制点连边,流量$2$,费用$0$: 控制点向两个方向的点各连两 ...

  4. bzoj AC倒序

    Search GO 说明:输入题号直接进入相应题目,如需搜索含数字的题目,请在关键词前加单引号 Problem ID Title Source AC Submit Y 1000 A+B Problem ...

  5. BZOJ 1096 【ZJOI2007】 仓库建设

    Description L公司有N个工厂,由高到底分布在一座山上.如图所示,工厂1在山顶,工厂N在山脚.由于这座山处于高原内陆地区(干燥少雨),L公司一般把产品直接堆放在露天,以节省费用.突然有一天, ...

  6. 【BZOJ 1096】 [ZJOI2007]仓库建设 (斜率优化)

    1096: [ZJOI2007]仓库建设 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 3940  Solved: 1736 Description ...

  7. bzoj 1096 仓库建设 -斜率优化

    L公司有N个工厂,由高到底分布在一座山上.如图所示,工厂1在山顶,工厂N在山脚.由于这座山处于高原内陆地区(干燥少雨),L公司一般把产品直接堆放在露天,以节省费用.突然有一天,L公司的总裁L先生接到气 ...

  8. 【BZOJ】1096: [ZJOI2007]仓库建设(dp+斜率优化)

    http://www.lydsy.com/JudgeOnline/problem.php?id=1096 首先得到dp方程(我竟然自己都每推出了QAQ)$$d[i]=min\{d[j]+cost(j+ ...

  9. BZOJ 2001: [Hnoi2010]City 城市建设

    2001: [Hnoi2010]City 城市建设 Time Limit: 20 Sec  Memory Limit: 162 MBSubmit: 1132  Solved: 555[Submit][ ...

随机推荐

  1. #Python编程从入门到实践#第二章笔记

      ​​​1.变量 (1)变量名只能包含字母.数字和下划线,不能包含空格 (2)不要将python关键字与函数名作为变量名 (3)简短有描述性,避免使用小写字母l和大写字母O (4)python 始终 ...

  2. Python学习笔记:logging(日志处理)

    在一个软件中,日志是可以说必不可少的一个组成部分,通常会在定位客户问题或者记录软件使用情况等场景中会用到.logging模板块是Python的一个内置标准库,用于实现对日志的控制输出,对于平常的日志输 ...

  3. python爬虫-简单使用xpath下载图片

    首先 1.为方便以下进行 谷歌浏览器里要安装xpath脚本 2.下载一个lmxl     命令:pip install lxml 3. 以下三张图是一个,当时爬的 <糗事百科>里的图片 值 ...

  4. 霍夫圆检测 opencv

    进行霍夫圆变换中有一个API:HoughCircles(). 第五个参数为double类型的minDist(),为霍夫变换检测到的圆的圆心之间的最小距离,即让算法能明显区分的两个不同圆之间的最小距离. ...

  5. python——闰年的判断

    写一个程序,判断给定年份是否为闰年. 这样定义闰年的:能被4整除但不能被100整除,或者能被400整除都是闰年. while(1): year = input("请输入一个年份,让我判断一下 ...

  6. K-均值聚类——电影类型

    K-均值聚类 K-均值算法试图将一系列样本分割成K个不同的类簇(其中K是模型的输入参数),其形式化的目标函数称为类簇内的方差和(within cluster sum of squared errors ...

  7. 关于 JS 模块化的最佳实践总结

    模块化开发是 JS 项目开发中的必备技能,它如同面向对象.设计模式一样,可以兼顾提升软件项目的可维护性和开发效率. 模块之间通常以全局对象维系通讯.在小游戏中,GameGlobal 是全局对象.在小程 ...

  8. 剑指Offer - 九度1510 - 替换空格

    剑指Offer - 九度1510 - 替换空格2013-11-29 20:53 题目描述: 请实现一个函数,将一个字符串中的空格替换成“%20”.例如,当字符串为We Are Happy.则经过替换之 ...

  9. 《Cracking the Coding Interview》——第3章:栈和队列——题目1

    2014-03-18 03:19 题目:用一个数组实现3个栈. 解法: 首先我想过让三个栈动态决定长度.要么左右各一个向中间靠拢,要么三个穿插着,后来都觉得实现起来太复杂,而且思路总有各种功能缺陷,会 ...

  10. leetcode 【 Intersection of Two Linked Lists 】python 实现

    题目: Write a program to find the node at which the intersection of two singly linked lists begins. Fo ...