2014 多校的B题,由于我不怎么搞图论,当时碰到这个题目,我怎么想都没往网络流方面弄,不过网络流真的是个好东西,对于状态多变,无法用动规或者数据结构来很好表示的时候,非常有用

这个题目要求每个点一定要访问到,并且每次访问的是没访问过的点,跳跃的方向为向右或者向下。

建图的时候,分成二分图,从一个超级源点向x部分建cap为1 cost为0的点,对所以可到达的点从x到y建cap为1,cost根据题目算出来,不过要算负值,因为我们要求得实际是最大费用,最后对结果求相反数即可。所有y部分的点对超级汇点T建cap为1,cost为0的点。

但是这样还是没满足只能走k次的条件,所以得额外建一个点,从S到该点建cap为k,cost为0的点,再用这个点对y部分所有点建cap为1,cost为0的点,因为每次从x部流到y部最终流到T的点,都是不包括起始点的,用这个额外点向y部分供应最多k次(不一定要k个)的起始点量即可

最后求得一定要流量为 N*M才算走通,此外,cost的相反数即为所求最大费用

用的是大白书上的最小费用最大模板,复杂度应该是n*m*某个系数

#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <vector>
#include <queue>
using namespace std;
const int maxn = 310;
struct Edge
{
int from,to,cap,flow,cost;
};
struct MCMF
{
int n,m,s,t;
vector<Edge> edges;
vector<int> G[maxn];
int inq[maxn];
int d[maxn];
int p[maxn];
int a[maxn];
void init(int n){
this->n=n;
for (int i=0;i<=n;i++) G[i].clear();
edges.clear();
}
void add(int from,int to,int cap,int cost)
{
edges.push_back((Edge){from,to,cap,0,cost});
edges.push_back((Edge){to,from,0,0,-cost});
m=edges.size();
G[from].push_back(m-2);
G[to].push_back(m-1);
}
} TF;
char mat[15][15];
int N,M,K;
int id[15][15];
int S,T;
bool Bellman(int s,int t,int &flow,int &cost)
{
for (int i=0;i<=2*N*M+2;i++) TF.d[i]=(1<<30);
memset(TF.inq,0,sizeof TF.inq);
TF.d[s]=0;TF.inq[s]=1;TF.a[s]=1<<30;
queue<int> Q;
Q.push(s);
while (!Q.empty())
{
int u=Q.front();Q.pop();
TF.inq[u]=0;
for (int i=0;i<TF.G[u].size();i++){
Edge & e=TF.edges[TF.G[u][i]];
if (e.cap>e.flow && TF.d[e.to]>TF.d[u]+e.cost){
TF.d[e.to]=TF.d[u]+e.cost;
TF.p[e.to]=TF.G[u][i];
TF.a[e.to]=min(TF.a[u],e.cap-e.flow);
if (!TF.inq[e.to]) {Q.push(e.to);TF.inq[e.to]=1;}
}
}
}
if (TF.d[t]>=(1<<30)) return false;
flow+=TF.a[t];
cost+=TF.d[t];
int u=t;
while (u!=s){
TF.edges[TF.p[u]].flow+=TF.a[t];
TF.edges[TF.p[u]^1].flow-=TF.a[t];
u=TF.edges[TF.p[u]].from;
}
return true;
}
int mincost()
{
int flow=0,cost=0;
while (Bellman(S,T,flow,cost));
if (flow==N*M)
return -cost;
else return -1;
}
int main()
{
int t;
scanf("%d",&t);
int kase=0;
while (t--)
{
scanf("%d%d%d",&N,&M,&K);
S=0,T=2*N*M+2;
TF.init(T);
for (int i=1;i<=N;i++) scanf("%s",mat[i]+1);
for (int i=1;i<=N;i++){
for (int c=1;c<=M;c++){
for (int j=i+1;j<=N;j++){
int cost=0;
cost+=j-i-1;
if (mat[i][c]==mat[j][c]) cost-=mat[i][c]-'0';
TF.add((i-1)*M+c,N*M+(j-1)*M+c,1,cost);
}
for (int j=c+1;j<=M;j++){
int cost=0;
cost+=j-c-1;
if (mat[i][c]==mat[i][j]) cost-=mat[i][c]-'0';
TF.add((i-1)*M+c,N*M+(i-1)*M+j,1,cost);
}
}
}
for (int i=1;i<=N;i++){
for (int j=1;j<=M;j++){
TF.add(S,(i-1)*M+j,1,0);
TF.add(N*M+(i-1)*M+j,T,1,0);
}
}
TF.add(S,2*N*M+1,K,0);
for (int i=N*M+1;i<=2*N*M;i++){
TF.add(2*N*M+1,i,1,0);
}
printf("Case %d : %d\n",++kase,mincost());
}
return 0;
}

  

HDU 4862 JUMP 最小费用最大流的更多相关文章

  1. hdu 2686 Matrix 最小费用最大流

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2686 Yifenfei very like play a number game in the n*n ...

  2. hdu 4494 Teamwork 最小费用最大流

    Teamwork Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://acm.hdu.edu.cn/showproblem.php?pid=4494 ...

  3. POJ 2195 Going Home / HDU 1533(最小费用最大流模板)

    题目大意: 有一个最大是100 * 100 的网格图,上面有 s 个 房子和人,人每移动一个格子花费1的代价,求最小代价让所有的人都进入一个房子.每个房子只能进入一个人. 算法讨论: 注意是KM 和 ...

  4. HDU 4862 Jump(更多的联合培训学校1)(最小费用最大流)

    职务地址:pid=4862">HDU4862 最小费用流做的还是太少. 建图想不出来. . . 直接引用官方题解的话吧... 最小K路径覆盖的模型.用费用流或者KM算法解决,构造二部图 ...

  5. HDU 4862 Jump 费用流

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

  6. hdu 1533 Going Home 最小费用最大流

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1533 On a grid map there are n little men and n house ...

  7. HDU 5988.Coding Contest 最小费用最大流

    Coding Contest Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)To ...

  8. hdu 3667(拆边+最小费用最大流)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3667 思路:由于花费的计算方法是a*x*x,因此必须拆边,使得最小费用流模板可用,即变成a*x的形式. ...

  9. hdu 2686&&hdu 3376(拆点+构图+最小费用最大流)

    Matrix Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Subm ...

随机推荐

  1. java打包成可执行的jar或者exe的详细步骤

    Java程序完成以后,对于Windows操作系统,习惯总是想双击某个exe文件就可以直接运行程序,现我将一步一步的实现该过程.最终结果是:不用安装JRE环境,不用安装数据库,直接双击一个exe文件,就 ...

  2. wxPython--学习笔记

    wxPython程序由两个必要的对象组成,应用对象APP和顶级窗口对象Frame 应用程序对象APP管理主事件循环MainLoop() 顶级窗口对象Frame管理数据,控制并呈现给用户 先看一段最简单 ...

  3. 2016 黑客必备的Android应用都有哪些?

    免责声明:本站所发布的此份清单仅供学习之用.我们不支持读者利用其中的任何工具进行任何不道德的恶意攻击行为. 根据业界的一系列评测以及亲身经验,我们整理出了这份最佳Android黑客应用清单.除了对应用 ...

  4. (5)LoraWAN:Join procedure、Receive Windows

    网络在建立之初,终端设备启动后需要向服务端发起Jion请求(接入请求),只有在接入请求得到成功答复,并根据答复配置相关参数后,终端才算成功加入网络.Jion成功后才能进行数据的上行.下行通信. Jio ...

  5. PyCharm 2018.1破解激活步骤

    1.下载破解补丁 下载地址:https://pan.baidu.com/s/1qjI9uHaw0x374rwu6H8djA 将下载下来的破解补丁放到C:\software\JetBrains\PyCh ...

  6. 【Unity】双击物体

    using UnityEngine; using System.Collections; using System; public class Click_yushe : MonoBehaviour ...

  7. SciPy 教程

    章节 SciPy 介绍 SciPy 安装 SciPy 基础功能 SciPy 特殊函数 SciPy k均值聚类 SciPy 常量 SciPy fftpack(傅里叶变换) SciPy 积分 SciPy ...

  8. 2017 青岛网络赛 Chenchen, Tangtang and ZengZeng

    Chenchen, Tangtang and ZengZeng are starting a game of tic-tac-toe, played on a 3 × 3 board. Initial ...

  9. Centos7 rsync+inotify两台服务器同步文件(单向)

    注:本篇介绍的是单向同步,即A文件同步到B,但B的文件不同步到A,双向同步的在下一篇文章中. rsync与inotify不再赘述,直接进入实战. 0.背景 两台服务器IP地址分别为: 源服务器:192 ...

  10. 文本处理三剑客与shell正则表达式

    文本处理三剑客 提到对于文本的处理上,除了vim这个强大的编辑器之外,还有使用命令的形式去处理你要处理的文本,而不需要手动打开文本再去编辑.这样做的好处是能够以shell命令的形式将编辑和处理文本的工 ...