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. Android 进程和线程模型

    Android进程模型 在安装Android应用程序的时候,Android会为每个程序分配一个Linux用户ID,并设置相应的权限,这样其它应用程序就不能访问此应用程序所拥有的数据和资源了. 在 Li ...

  2. DRP过后,感受知识间的通性

    DRP视频看了不短的时间,真正开始DRP的时间是7月17号,至今两个月了.由于暑假期间英语的学习占得时间比较多,所以DRP视频进行的很慢.9月11号看完了DRP所有的视频,这个项目完成后最大的感受是: ...

  3. JAVA ThreadPoolExecutor(转)

    原文链接:http://blog.csdn.net/historyasamirror/article/details/5961368 基础 在我看来,java比C++的一个大好处就是提供了对多线程的支 ...

  4. android 支持分组和联系人展示的一个小样例

    先看效果图: 要实现这个效果,activity必须实现ExpandableListActivity @Override public void onCreate(Bundle savedInstanc ...

  5. Python lambda和reduce函数

    看到一篇博文写lambda和reduce函数.笔者小痒了一下,用Python实现一下: #! /usr/bin/env python # -*-coding:utf-8-*- import time ...

  6. Aspx 页面生命周期

    ASP.NET 页运行时,此页将经历一个生命周期,在生命周期中将执行一系列处理步骤.这些步骤包括初始化.实例化控件.还原和维护状态.运行事件处理程序代码以及进行 呈现.了解页的生命周期非常重要,这样就 ...

  7. if语句之猜拳

    用计算机来生成随机数: Random rand = new Random();//做一个随机生成器,Random();后面的括号里面可以放一个随机生成器种子,这个种子只能为整数(int)int n = ...

  8. 简单字符串模式匹配算法的C++实现

    /* * simpleIndex.cpp * Author: Qiang Xiao * Time: 2015-07-13 */ #include<iostream> #include< ...

  9. Tortoisegit 记住用户名和密码

    Tortoisegit 记住用户名和密码方法: [Windows系统] 当你配置好git后,在 C:\Documents and Settings\Administrator\ 目录下有一个  .gi ...

  10. linq 总结

    linq 常用方法: top   var query=(from u in User ...).Take(10) dblinq的坑: 时间必须当参数传入,否则会报错 多个left join时,如果jo ...