POJ 3422 Kaka's Matrix Travels


题意:有一个N*N的方格,每一个方格里面有一个数字。如今卡卡要从左上角走到右下角,规定每次仅仅能向下或者向右走。每次走到一个格子,将得到该格子的数字,而且该格子的数字变为0。当卡卡走一次时,非常easy求出最大值,问卡卡走k次,可以得到的最大值为多少。

思路:最小费用最大流
关键是怎样构图
1. 将N*N个格点拆分为两个点(i,i + N*N),每一个点之间连一条流量为1,费用为-w的边。再连一条流量为k,费用为0的边。这样就保证了每一个点之间能够走k次。且最多仅仅有一次能拿到费用。
2. 将每一个点与其以下、右边的点连边。流量为k。费用为0.
3. 构造两个源点、汇点。

源点和左上角连边,流量为k,费用为0. 右下角与汇点连边。流量为k,费用为0.


至此,容量网络已经构成。

每次在残余网络中找费用最短路进行增广就可以。


代码:
/*
ID: wuqi9395@126.com
PROG:
LANG: C++
*/
#include<map>
#include<set>
#include<queue>
#include<stack>
#include<cmath>
#include<cstdio>
#include<vector>
#include<string>
#include<fstream>
#include<cstring>
#include<ctype.h>
#include<iostream>
#include<algorithm>
using namespace std;
#define INF (1 << 20)
#define LINF (1LL << 60)
#define PI acos(-1.0)
#define mem(a, b) memset(a, b, sizeof(a))
#define rep(i, a, n) for (int i = a; i < n; i++)
#define per(i, a, n) for (int i = n - 1; i >= a; i--)
#define eps 1e-6
#define debug puts("===============")
#define pb push_back
#define mkp make_pair
#define all(x) (x).begin(),(x).end()
#define fi first
#define se second
#define SZ(x) ((int)(x).size())
#define POSIN(x,y) (0 <= (x) && (x) < n && 0 <= (y) && (y) < m)
typedef long long ll;
typedef unsigned long long ULL;
const int maxn = 5555;
const int maxm = 500000;
struct node {
int v, cap, nxt, cost;
} e[maxm * 2];
int g[maxn], cnt, st, ed, n, m;
int ans, flow;
int nt, k;
void add(int u, int v, int cap, int cost) {
e[++cnt].v = v;
e[cnt].cap = cap;
e[cnt].cost = cost;
e[cnt].nxt = g[u];
g[u] = cnt; e[++cnt].v = u;
e[cnt].cap = 0;
e[cnt].cost = -cost;
e[cnt].nxt = g[v];
g[v] = cnt;
}
void init() {
cnt = 1;
ans = flow = 0;
memset(g, 0, sizeof(g));
// 加边
int w;
int p = nt * nt;
for (int i = 1; i <= nt; i++) {
for (int j = 1; j <= nt; j++) {
scanf("%d", &w);
int id = (i - 1) * nt + j;
add(id, id + p, 1, -w);
add(id, id + p, k, 0);
if (i < nt) add(id + p, id + nt, k, 0);
if (j < nt) add(id + p, id + 1, k, 0);
}
}
st = 0, ed = p * 2 + 1;
n = ed;
add(st, 1, k, 0);
add(p * 2, ed, k, 0);
} int dis[maxn], que[maxn], pre[maxn];
bool vis[maxn];
bool spfa() {
int font = 0, rear = 1;
for(int i = 0; i <= n; i ++) {
dis[i] = INF;
vis[i] = false;
}
dis[st] = 0;
que[0] = st;
vis[st] = true;
while(rear != font) {
int u = que[font++];
font %= n;
vis[u] = false;
for(int i = g[u]; i; i = e[i].nxt) {
int v = e[i].v;
if(e[i].cap && dis[v] > dis[u] + e[i].cost) {
dis[v] = dis[u] + e[i].cost;
pre[v] = i;
if(!vis[v]) {
vis[v] = true;
que[rear++] = v;
rear %= n;
}
}
}
}
if(dis[ed] == INF) return false;
return true;
}
void augment() {
int u, p, mi = INF;
for(u = ed; u != st; u = e[p ^ 1].v) {
p = pre[u];
mi = min(mi, e[p].cap);
}
for(u = ed; u != st; u = e[p ^ 1].v) {
p = pre[u];
e[p].cap -= mi;
e[p ^ 1].cap += mi;
ans += mi * e[p].cost; // cost记录的为单位流量费用。必须得乘以流量。
}
flow += mi;
}
int MCMF() {
init();
while(spfa()) augment();
return ans;
}
int main () {
while(~scanf("%d%d", &nt, &k)) {
printf("%d\n", -MCMF());
}
return 0;
}

POJ 3422 Kaka&#39;s Matrix Travels (最小费用最大流)的更多相关文章

  1. POJ 3422 Kaka&#39;s Matrix Travels(费用流)

    POJ 3422 Kaka's Matrix Travels 题目链接 题意:一个矩阵.从左上角往右下角走k趟,每次走过数字就变成0,而且获得这个数字,要求走完之后,所获得数字之和最大 思路:有点类似 ...

  2. POJ训练计划3422_Kaka&#39;s Matrix Travels(网络流/费用流)

    解题报告 题目传送门 题意: 从n×n的矩阵的左上角走到右下角,每次仅仅能向右和向下走,走到一个格子上加上格子的数,能够走k次.问最大的和是多少. 思路: 建图:每一个格子掰成两个点,分别叫" ...

  3. [poj] 3422 Kaka's Matrix Travels || 最小费用最大流

    原题 给一个N*N的方阵,从[1,1]到[n,n]走K次,走过每个方格加上上面的数,然后这个格上面的数变为0.求可取得的最大的值. 要求最大值,所以把边权全为负跑最小费用即可.因为只有第一次经过该点的 ...

  4. POJ 2135 Farm Tour (网络流,最小费用最大流)

    POJ 2135 Farm Tour (网络流,最小费用最大流) Description When FJ's friends visit him on the farm, he likes to sh ...

  5. poj3422 Kaka's Matrix Travels(最小费用最大流问题)

    /* poj3422 Kaka's Matrix Travels 不知道 k次 dp做为什么不对??? 看了大牛的代码,才知道还可以这样做! 开始没有理解将a 和 a‘ 之间建立怎样的两条边,导致程序 ...

  6. POJ3422 Kaka&#39;s Matrix Travels 【最大费用最大流】

    Kaka's Matrix Travels Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 8006   Accepted:  ...

  7. POJ 3686:The Windy's(最小费用最大流)***

    http://poj.org/problem?id=3686 题意:给出n个玩具和m个工厂,每个工厂加工每个玩具有一个时间,问要加工完这n个玩具最少需要等待的平均时间.例如加工1号玩具时间为t1,加工 ...

  8. poj Kaka&#39;s Matrix Travels

    Kaka's Matrix Travels 题目: 给出一个矩阵.求仅仅能向下或者向右的情况下能得到的最大和.一般的是指遍历一次,而这个是能够反复走K次.每经过一次后就把该点设为0.求最大和. 算法: ...

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

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

随机推荐

  1. rpc的学习

    rpc(Remote process call 即远程过程调用)是一种请求-相应的协议, 主要使用于C/S架构中,使得分布式系统成为可能.由客户端发起请求,服务端调用各种参数处理请求,当服务器在处理请 ...

  2. flex正则表达式

    正则表达式是一种通用的标准,大部分计算机语言都支持正则表达式,包括as3,这里收集了一些常用的正则表达式语句,大家用到的时候就不用自己写了 ^\d+$ //匹配非负整数(正整数 + 0) ^[0-9] ...

  3. 发一个讨论帖,如果结果被采纳的话可以给一份adb 代码,以及我封装的ADBLIB

    如何在手机没有root 的情况下,获取系统的一些文件,比如 /data/data/xxxx 目录下的文件. 有任何想法的请说出来.

  4. ADO.NET基础笔记

    ADO.NET 程序要和数据库交互要通过ADO.NET进行,通过ADO.Net就能在程序中执行SQL了. ADO.Net中提供了对各种不同的数据库的统一操作接口. 连接字符串: 程序通过连接字符串指定 ...

  5. CodeForce 569A

    Time Limit:2000MS     Memory Limit:262144KB     64bit IO Format:%I64d & %I64u   Description Litt ...

  6. javascript学习(9)——[设计模式]单例

    单例模式,相信大家对此都不陌生,我们主要讲下javascript中几个比较常见的设计模式: (1).普通的单体 (2).具有局部变量的强大单体 (3).惰性单体 (4).分支单体 下面我们就一一进行介 ...

  7. Zend Studio 如何配置本地apache服务器使用xdebug调试php脚本

    本地环境搭配: apache 2.2 安装位置:D:/program files/Apache Software Foundation/Apache2.2 php 5.2.10 安装位置:C:/php ...

  8. Python 模块(八) socketserver 以及 线程、进程

    目录 异常处理 socketserver 线程.进程 一.异常处理 try的工作原理是,当开始一个try语句后,python就在当前程序的上下文中作标记,这样当异常出现时就可以回到这里,try子句先执 ...

  9. 无法更新 EntitySet“GuigeInfo”,因为它有一个 DefiningQuery,而 <ModificationFunctionMapping> 元素中没有支持当前操作的 <InsertFunction> 元素。

    1:实体中必须有主键 2:删除创建的模型重新创建

  10. STL string 模拟

    下面的代码来自c++ primer plus第5版第12章,书中代码写的非常好: // string1.h -- fixed and augmented string class definition ...