题意:

给定一个网格,每个网格有选取代价和占据收益。每个点被占据,需要满足以下两个条件至少一个条件:1.被选取  2.邻近方格都被选取(有公共边被称为邻近)  不一定要占据所有方格,求最大收益。

输入说明
第一行两个数 n,m(n,m ≤ 20),表示矩形的长和宽。
接下来 n 行,每行是 m 个字符组成的字符串,描述投资的花费。
接下来 n 行,每行是 m 个字符组成的字符串,表示该格子的收益。
花费和收益按照一种奇葩的方式给出:
字符

‘0’-’9’
0-9
‘a’-’z’
10-35
‘A’-’Z’
36-61
输出说明
一个数,表示收益的和减去投资的和的最大值。
样例 1
2 2
21
12
21
12
答案:4
样例 2
2 2
ZZ
ZZ
11
11
答案: 0
样例 3
3 3
XXX
XXX
XXX
aaa
aZa
aaa
答案: 2
样例 4
2 4
asam
atik
123A
45BC
答案: 71
样例 5
98
IIIIIIII
IIWWWWII
IIWIIIII
IIWIIIII
IIWWWWII
IIIIIWII
IIIIIWII
IIWWWWII
IIIIIIII
IIIIIIII
II0000II
II0II0II
II0II0II
II0000II
II0II0II
II0II0II
II0000II
IIIIIIII
答案: 606

先黑白染色

源点S点连白点,汇点T点连黑点

现将所有利润加起来,显然要减去一部分花费和利润

我们求最小的使利润合法的要减去一部分花费和利润

然后转化为求最小割,设收益边和花费边

首先考虑几种情况

1.保留一个点的花费边,保留收益边(即不投资拿到利润)

那么此刻如果相邻有点没有割掉花费边,那么就会有流

2.割掉一个花费边,保留收益边(投资一个点)

3.割掉一个收益边,保留花费边(不投资该点)

只考虑前2中情况我们可以建出:

考虑第3种情况:

因为放弃这个格子的收益,代表着相邻的格子的收益必须要靠它自己的花费

于是有了如下建图

图片转自ZYYS

 #include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
#include<queue>
using namespace std;
struct Node
{
int next,to,cap;
}edge[];
int head[],num=,inf=1e9,S,T,dist[],n,m,a[][],b[][],ans,tot,pre[][],nxt[][],col[][];
char s1[][],s2[][];
void add(int u,int v,int cap)
{
num++;
edge[num].next=head[u];
head[u]=num;
edge[num].to=v;
edge[num].cap=cap;
num++;
edge[num].next=head[v];
head[v]=num;
edge[num].to=u;
edge[num].cap=;
}
bool bfs()
{int i;
queue<int>Q;
for (i=S;i<=T;i++)
dist[i]=-;
dist[S]=;
Q.push(S);
while (Q.empty()==)
{
int u=Q.front();
Q.pop();
for (i=head[u];i;i=edge[i].next)
{
int v=edge[i].to;
if (edge[i].cap==||dist[v]!=-) continue;
dist[v]=dist[u]+;
Q.push(v);
}
}
if (dist[T]==-) return ;
return ;
}
int dfs(int x,int flow,int des)
{int i;
if (x==des) return flow;
if (flow<=) return ;
int res=,tmp;
for (i=head[x];i;i=edge[i].next)
{
int v=edge[i].to;
if (dist[v]==dist[x]+&&edge[i].cap)
{
tmp=dfs(v,min(edge[i].cap,flow-res),des);
edge[i].cap-=tmp;
edge[i^].cap+=tmp;
res+=tmp;
if (res==flow) return res;
}
}
return res;
}
int Maxflow()
{
int as=;
while (bfs())
{
int a=;
while (a=dfs(S,inf,T)) as+=a;
}
return as;
}
int main()
{int i,j;
cin>>n>>m;
for (i=;i<n;i++)
{
scanf("%s",s1[i]);
}
for (i=;i<n;i++)
{
scanf("%s",s2[i]);
}
for (i=;i<=n;i++)
{
for (j=;j<=m;j++)
{
if (s1[i-][j-]>=''&&s1[i-][j-]<='')
a[i][j]=s1[i-][j-]-'';
else if (s1[i-][j-]>='a'&&s1[i-][j-]<='z')
a[i][j]=s1[i-][j-]-'a'+;
else a[i][j]=s1[i-][j-]-'A'+;
}
}
for (i=;i<=n;i++)
{
for (j=;j<=m;j++)
{
if (s2[i-][j-]>=''&&s2[i-][j-]<='')
b[i][j]=s2[i-][j-]-'';
else if (s2[i-][j-]>='a'&&s2[i-][j-]<='z')
b[i][j]=s2[i-][j-]-'a'+;
else b[i][j]=s2[i-][j-]-'A'+;
ans+=b[i][j];
}
}
S=;tot=;
for (i=;i<=n;i++)
{
for (j=;j<=m;j++)
{
pre[i][j]=++tot;nxt[i][j]=++tot;
col[i][j]=(i+j)&;
}
}
T=++tot;
for (i=;i<=n;i++)
{
for (j=;j<=m;j++)
{
if (col[i][j]==)
{
add(S,pre[i][j],a[i][j]);
add(pre[i][j],nxt[i][j],b[i][j]);
if (i-)
add(pre[i][j],nxt[i-][j],inf),add(nxt[i][j],pre[i-][j],inf);
if (i<n)
add(pre[i][j],nxt[i+][j],inf),add(nxt[i][j],pre[i+][j],inf);
if (j-)
add(pre[i][j],nxt[i][j-],inf),add(nxt[i][j],pre[i][j-],inf);
if (j<m)
add(pre[i][j],nxt[i][j+],inf),add(nxt[i][j],pre[i][j+],inf);
}
else
{
add(pre[i][j],T,a[i][j]);
add(nxt[i][j],pre[i][j],b[i][j]);
}
}
}
cout<<ans-Maxflow();
}

SRM 558 SurroundingGame的更多相关文章

  1. TopCoder SRM 558 Div 1 - Problem 1000 SurroundingGame

    传送门:https://284914869.github.io/AEoj/558.html 题目简述  一个人在一个n * m棋盘上玩游戏,想要占领一个格子有两个方法: 在这个格子放一个棋子.  这个 ...

  2. 记第一次TopCoder, 练习SRM 583 div2 250

    今天第一次做topcoder,没有比赛,所以找的最新一期的SRM练习,做了第一道题. 题目大意是说 给一个数字字符串,任意交换两位,使数字变为最小,不能有前导0. 看到题目以后,先想到的找规律,发现要 ...

  3. SRM 513 2 1000CutTheNumbers(状态压缩)

    SRM 513 2 1000CutTheNumbers Problem Statement Manao has a board filled with digits represented as St ...

  4. SRM 510 2 250TheAlmostLuckyNumbersDivTwo(数位dp)

    SRM 510 2 250TheAlmostLuckyNumbersDivTwo Problem Statement John and Brus believe that the digits 4 a ...

  5. UVA 558 判定负环,spfa模板题

    1.UVA 558 Wormholes 2.总结:第一个spfa,好气的是用next[]数组判定Compilation error,改成nexte[]就过了..难道next还是特殊词吗 题意:科学家, ...

  6. SRM 657 DIV2

    -------一直想打SRM,但是感觉Topcoder用起来太麻烦了.题目还是英文,不过没什么事干还是来打一打好了.但是刚注册的号只能打DIV2,反正我这么弱也只适合DIV2了.. T1: 题目大意: ...

  7. uva 558 - Wormholes(Bellman Ford判断负环)

    题目链接:558 - Wormholes 题目大意:给出n和m,表示有n个点,然后给出m条边,然后判断给出的有向图中是否存在负环. 解题思路:利用Bellman Ford算法,若进行第n次松弛时,还能 ...

  8. SRM DIV1 500pt DP

    SRM 501 DIV1 500pt SRM 502 DIV1 500pt SRM 508 DIV1 500pt SRM 509 DIV1 500pt SRM 511 DIV1 500pt SRM 5 ...

  9. TC srm 673 300 div1

    TC srm.673 300 Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 Description 给你n(n<=50)匹马和n个人,一匹马和一个人能 ...

随机推荐

  1. g第十四周,十五周作业

    1.数组中偶数的和 #include <stdio.h> int main(){ ; ]; ;i<=;i++) { scanf("%d ",&a[i]); ...

  2. 201621123062《java程序设计》第九周作业总结

    1. 本周学习总结 1.1 以你喜欢的方式(思维导图或其他)归纳总结集合与泛型相关内容. 思维导图1:(对集合部分做了一些改动和细化) 思维导图2:(泛型) 1.2 选做:收集你认为有用的代码片段 代 ...

  3. Beta冲刺 第二天

    Beta冲刺 第二天 1. 昨天的困难 由于前面的冲刺留下的问题很多,而且混乱的代码给我们接下来的完善工作带来了巨大的困难. 2. 今天解决的进度 潘伟靖: 1.对代码进行了review 2.为系统增 ...

  4. pandas 数据分析使用

    https://github.com/Erick-LONG/data_analysis/blob/master/%E6%95%B0%E6%8D%AE%E5%88%86%E6%9E%90%20%E9%8 ...

  5. OO前三次作业总结

    一.第一次作业 1.程序设计分析 ![img](s1.ax1x.com/2018/04/02/CSgoSU.png) 图1 第一次作业类图 ![name](https://images2018.cnb ...

  6. 深入浅出 SSL 管理配置实战

    我们生活在一个信息大爆炸的时代,几乎每天都在和互联网打交道,购物.网银转账.支付宝付款.搜索信息.查看邮件.观看视频.微信聊天.上网冲浪.阅读新闻等,无不时时刻刻在和网络打交道.那如何保护网络安全就相 ...

  7. JS 转换数据类型

    JavaScript 是一种动态数据类型语言,变量是没有类型的,可以随机赋予任意值,若变量要转换数据类型,有两种办法:隐式转换和显式转换. 隐式转换可转换为字符串(将一个值加上字符串) 数字(在值的前 ...

  8. Java面试题合集(二)

    接下来几篇文章准备系统整理一下有关Java的面试题,分为基础篇,javaweb篇,框架篇,数据库篇,多线程篇,并发篇,算法篇等等,陆续更新中.其他方面如前端后端等等的面试题也在整理中,都会有的. 注: ...

  9. centos 6.5安装并配置mysql

    折腾了半天终于把mysql安装并配置好了,以下是安装步骤和遇到问题的解决方式 1.查看机器上是否已经安装了mysql或其相关项 # yum list installed | grep mysql如果安 ...

  10. System.Reflection名称空间下的程序集类Assembly应用.

    利用反射中的程序集类(Assembly--抽象类)动态加载类库(.dll)或者可执行程序(.exe). 优点:①.可以消除if条件的逻辑判断.②.减少内存资源.③.有利于程序扩展. 缺点... 使用静 ...