题意:

      给你一个矩阵,你最多可以选择k条路线,k条路线的起点随意,每次行走的距离随意,但是只能往右或者下走,走过的点不能再走,而且每一步如果a->b,如果a和b的权值s相等那么就可以获得s这么多能量,每一步花费的能量是|x1 - x2| + |y1 - y2| - 1,最后问你吧所有点都遍历后获得的最大能量是多少。

思路:

      每个点只能走一次,最多可以选择k条路径,要求全走完的最大,显然是费用流,我写的是最大费用最大流,(最小费用最大流也行,就是正负的问题)下面说建边。

  (1) 首先虚拟出来三个点,源点s,终点e,超级远点ss。

 *(2) 拆点 a -> a'  流量1,费用INF //每个点只能访问一次,INF是为了让所有的点都访 问到,这个是关键,如果     是最消费用流的话可以使 -INF 

  (3) ss -> s 费用0流量k  //限制路径条数

  (4) s -> a  费用0流量1  // 每一个点都可以当路径起点

  (5) a' -> b 费用 可获得价值-花费 流量是1 // 每个点让他右和下的所有点建边

  (6) a' -> e 费用0,流量1 //每个点都可以做路径终点

 *(7) s -> e  费用0,流量INF // 题目说k可以不全用,排除强制用k条导致的错误

      这样就行了,还有就是说明下这个题目如果可以覆盖的话根本不存在负的答案,题目的那句话是误导人的,如果出现负的,那么就直接是无解的,因为ans = mclow - n * m * INF,INF很大,只要有一个格子没被覆盖过,那么abs就直接是负值了。



#include<stdio.h>
#include<string.h>
#include<queue> #define N_node 220
#define N_edge 33000
#define INF 10000000

using namespace
std; typedef struct
{
int
from ,to ,next ,cost ,flow;
}
STAR; STAR E[N_edge];
int
list[N_node] ,tot;
int
s_x[N_node];
int
mer[N_edge]; void add(int a ,int b ,int c ,int d)
{

E[++tot].from = a;
E[tot].to = b;
E[tot].cost = c;
E[tot].flow = d;
E[tot].next = list[a];
list[a] = tot; E[++tot].from = b;
E[tot].to = a;
E[tot].cost = -c;
E[tot].flow = 0;
E[tot].next = list[b];
list[b] = tot;
} bool
spfa(int s ,int t ,int n)
{
for(int
i = 0 ;i <= n ;i ++)
s_x[i] = -1000000000;
int
mark[N_node] = {0};
s_x[s] = 0 ,mark[s] = 1;
queue<int>q;
q.push(s);
memset(mer ,255 ,sizeof(mer));
while(!
q.empty())
{
int
xin ,tou;
tou = q.front();
q.pop();
mark[tou] = 0;
for(int
k = list[tou] ;k ;k = E[k].next)
{

xin = E[k].to;
if(
E[k].flow && s_x[xin] < s_x[tou] + E[k].cost)
{

mer[xin] = k;
s_x[xin] = s_x[tou] + E[k].cost;
if(!
mark[xin])
{

mark[xin] = 1;
q.push(xin);
}
}
}
}
return
mer[t] != -1;
} int
M_C_Flow(int s ,int t ,int n ,int nn ,int mm)
{
int
maxcost = 0 ,minflow ,maxflow = 0;
while(
spfa(s ,t ,n))
{

maxflow = INF;
for(int
i = mer[t] ;i + 1 ;i = mer[E[i].from])
{
if(
minflow > E[i].flow)
minflow = E[i].flow;
}
for(int
i = mer[t] ;i + 1 ;i = mer[E[i].from])
{

E[i].flow -= minflow;
E[i^1].flow += minflow;
maxcost += E[i].cost * minflow;
}

maxflow += minflow;
}
return
maxcost;
} int
abss(int x)
{
return
x < 0 ? -x : x;
} int main ()
{
int
map[15][15];
int
n ,m ,k ,i ,cas = 1 ,t ,j;
scanf("%d" ,&t);
while(
t--)
{

scanf("%d %d %d" ,&n ,&m ,&k);
char
tmp[15];
for(
i = 1 ;i <= n ;i ++)
{

scanf("%s" ,tmp);
for(
j = 1 ;j <= m ;j ++)
map[i][j] = tmp[j-1] - '0';
}

memset(list ,0 ,sizeof(list)) ,tot = 1; for(i = 1 ;i <= n ;i ++)
for(
j = 1 ;j <= m ;j ++)
{
int
now = (i - 1) * m + j;
add(now ,now + n * m ,INF ,1);
} int
s = 0 ,ss = n * m * 2 + 1 ,e = n * m * 2 + 2;
add(s ,ss ,0 ,k);
for(
i = 1 ;i <= n ;i ++)
for(
j = 1 ;j <= m ;j ++)
{
int
now = (i - 1) * m + j;
add(ss ,now ,0 ,1);
add(now + n * m ,e ,0 ,1);
for(
k = i + 1 ;k <= n ;k ++)
{
int
to = (k - 1) * m + j;
int
dis = abss(i - k) - 1;
if(
map[i][j] == map[k][j])
add(now + n * m ,to ,map[i][j] - dis,1);
else
add(now + n * m ,to ,0 - dis ,1);
} for(
k = j + 1 ;k <= m ;k ++)
{
int
to = (i - 1) * m + k;
int
dis = abss(j - k) - 1;
if(
map[i][j] == map[i][k])
add(now + n * m ,to ,map[i][j] - dis,1);
else
add(now + n * m ,to ,0 - dis ,1);
}
}

add(ss ,e ,0 ,INF);
int
ans = M_C_Flow(s ,e ,e ,n ,m) - n * m * INF;
if(
ans < 0) ans = -1;
printf("Case %d : " ,cas ++);
printf("%d\n" ,ans);
}
return
0;
}

hdu4862 费用流(不错)的更多相关文章

  1. HDU4807 Lunch Time(费用流变种)

    题目 Source http://acm.hdu.edu.cn/showproblem.php?pid=4807 Description The campus of Nanjing Universit ...

  2. 网络费用流-最小k路径覆盖

    多校联赛第一场(hdu4862) Jump Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Ot ...

  3. bzoj3876: [Ahoi2014&Jsoi2014]支线剧情(上下界费用流)

    传送门 一道题让我又要学可行流又要学zkw费用流…… 考虑一下,原题可以转化为一个有向图,每次走一条路径,把每一条边都至少覆盖一次,求最小代价 因为一条边每走过一次,就要付出一次代价 那不就是费用流了 ...

  4. 洛谷P3381 【模板】最小费用最大流(dijstra费用流)

    题目描述 如题,给出一个网络图,以及其源点和汇点,每条边已知其最大流量和单位流量费用,求出其网络最大流和在最大流情况下的最小费用. 输入输出格式 输入格式: 第一行包含四个正整数N.M.S.T,分别表 ...

  5. HDU 4862 Jump 费用流

    又是一个看了题解以后还坑了一天的题…… 结果最后发现是抄代码的时候少写了一个负号. 题意: 有一个n*m的网格,其中每个格子上都有0~9的数字.现在你可以玩K次游戏. 一次游戏是这样定义的: 你可以选 ...

  6. hdu-5988 Coding Contest(费用流)

    题目链接: Coding Contest Time Limit: 2000/1000 MS (Java/Others)     Memory Limit: 65536/65536 K (Java/Ot ...

  7. POJ2195 Going Home[费用流|二分图最大权匹配]

    Going Home Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 22088   Accepted: 11155 Desc ...

  8. BZOJ3130: [Sdoi2013]费用流[最大流 实数二分]

    3130: [Sdoi2013]费用流 Time Limit: 10 Sec  Memory Limit: 128 MBSec  Special JudgeSubmit: 960  Solved: 5 ...

  9. 洛谷 1004 dp或最大费用流

    思路: dp方法: 设dp[i][j][k][l]为两条没有交叉的路径分别走到(i,j)和(k,l)处最大价值. 则转移方程为 dp[i][j][k][l]=max(dp[i-1][j][k-1][l ...

随机推荐

  1. 中小企业上云首选,华为云全新云服务器S6性能评测分析

    转: 中小企业上云首选,华为云全新云服务器S6性能评测分析 [小宅按]今天,华为云全新弹性云服务器ECS通用计算型云服务器S6(以下简称为"华为云S6云服务器")正式发布,至顶网云 ...

  2. MAC (Message Authentication Code,消息认证码算法)

    需要将密钥发送到对方,对方用该密钥进行摘要处理,进行摘要验证. //初始化KeyGenerator KeyGenerator keyGenerator= KeyGenerator.getInstanc ...

  3. Codeforces Round #545 B. Circus

    题面: 传送门 题目描述: 马戏团中一共有N个人(N是偶数),有的人会扮演小丑,有的人会表演杂技.给出每个人会什么,然后按照下列规则把这些人分成两组: 每个人只能在其中一组 两个组的人数相等(也就是把 ...

  4. maven 常用命名

    maven项目,在命令行中操作,非常简洁.高效,现将maven项目常用命令行总结如下: maven命令行命令总结 序号 整理 统计 命令 作用 1 基本 5 mvn -v 查看maven版本 2 mv ...

  5. webpack核心模块tapable用法解析

    前不久写了一篇webpack基本原理和AST用法的文章,本来想接着写webpack plugin的原理的,但是发现webpack plugin高度依赖tapable这个库,不清楚tapable而直接去 ...

  6. P1601_A+B Problem(高精)(JAVA语言)

    思路:BigInteger first blood! //四行搞定 题目背景 无 题目描述 高精度加法,x相当于a+b problem,[b][color=red]不用考虑负数[/color][/b] ...

  7. WERTYU_键盘错位(JAVA语言)

    package 第三章; import java.util.Scanner; /*  *  把手放在键盘上时,稍不注意就会往右错一位.这样,输入Q会变成输入W,输入J会变成输入K等.        输 ...

  8. python-3-2

    一 切片 1.切片是list取值的一种方式 列子: nums = ['a','b','c','d','e','f','h','g','k','l','kk','nn','ee'] 取b和c元素出来 p ...

  9. C语言中复杂声明的解读和简化

    code[class*="language-"], pre[class*="language-"] { color: rgba(51, 51, 51, 1); ...

  10. 没想到吧,Java开发 API接口可以不用写 Controller了

    本文案例收录在 https://github.com/chengxy-nds/Springboot-Notebook 大家好,我是小富~ 今天介绍我正在用的一款高效敏捷开发工具magic-api,顺便 ...