题意:

      给你一个矩阵,你最多可以选择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. POJ-3436(网络流+最大流+输出路径)

    ACM Computer Factory POJ-3436 题目就是一个工厂n个加工机器,每个机器有一个效率w,q个材料入口,q个材料出口,每个口有三个数表示状态,1表示一定有入/出的材料,0表示没有 ...

  2. C++树——遍历二叉树

    在讲遍历之前,我们要先创建一个树: #include <iostream> using namespace std; typedef struct node; typedef node * ...

  3. Python爬虫学习三------requests+BeautifulSoup爬取简单网页

    第一次第一次用MarkDown来写博客,先试试效果吧! 昨天2018俄罗斯世界杯拉开了大幕,作为一个伪球迷,当然也得为世界杯做出一点贡献啦. 于是今天就编写了一个爬虫程序将腾讯新闻下世界杯专题的相关新 ...

  4. C# 基础 - 文件对话框

    using System.Windows.Forms; ... /// <summary> /// 选择保存文件的名称以及路径 取消返回 空""; /// </s ...

  5. Java流程控制:用户交互Scanner

    java.util.Scanner工具类获取用户输入语法:Scanner scanner = new Scanner(System.in);通过Scanner类的next()与nextLine()方法 ...

  6. 图文详解:内存总是不够,我靠HBase说服了Leader为新项目保驾护航

  7. Spring如何解决循环依赖

    一.什么是循环依赖 多个bean之间相互依赖,形成了一个闭环. 比如:A依赖于B.B依赖于c.c依赖于A 通常来说,如果问spring容器内部如何解决循环依赖, 一定是指默认的单例Bean中,属性互相 ...

  8. java网络通信不止UDP,TCP

    预备知识 多线程 实现多线程 线程池 IO流 核心功能就是读和写 扩展功能对什么读写,怎么读写,如何优化读写 网络基础 IP IP规定网络上所有的设备都必须有一个独一无二的IP地址,就好比是邮件上都必 ...

  9. Everything about WSL 1 you want to know

    关于 WSL 1 入门,你应该知道这些 如有错误,欢迎指出 参考: WSL 文档 VMware Workstation Pro 文档 概述 通过 WSL 2 来认识 WSL 1 什么是 WSL 2? ...

  10. Linux内核模块驱动加载与dmesg调试

    因为近期用到了Linux内核的相关知识,下面随笔将给出内核模块的编写记录,供大家参考. 1.运行环境 Ubuntu 版本:20.04 Linux内核版本:5.4.0-42-generic gcc版本: ...