Fork me on GitHub

【luogu4474王者之剑】--网络流

题目描述

https://cdn.luogu.com.cn/upload/pic/17920.png
这是在阿尔托利亚·潘德拉贡成为英灵前的事情,她正要去拔出石中剑成为亚瑟王,在这之前她要去收集一些宝石。
宝石排列在一个n*m的网格中,每个网格中有一块价值为v(i,j)的宝石,阿尔托利亚·潘德拉贡可以选择自己的起点。
开始时刻为0秒。以下操作,每秒按顺序执行

  1. 在第i秒开始的时候,阿尔托利亚·潘德拉贡在方格(x,y)上,她可以拿走(x,y)中的宝石。
  2. 在偶数秒,阿尔托利亚·潘德拉贡周围四格的宝石会消失
  3. 若阿尔托利亚·潘德拉贡第i秒开始时在方格(x,y)上,则在第i+1秒可以立即移动到(x+1,y),(x,y+1),(x-1,y)或(x,y-1)上,也可以停留在(x,y)上。

求阿尔托利亚·潘德拉贡最多可以获得多少价值的宝石

输入格式

第一行给出数字N,M代表行列数.N,M均小于等于100,宝石的价值不会超过10000.下面N行M列用于描述数字矩阵

输出格式

输出最多可以拿到多少价值宝石

输入输出样例

输入 #1
2 2
1 2
2 1
输出 #1
4

说明/提示

姚金宇的原创题。

代码:

include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<queue>
#define R register
#define inf 1e9+7

using namespace std;
const int Excalibur = 10005;//N或MAXN
int n, m, S, T, cur[Excalibur], dep[Excalibur];
struct saber {//edge,我的王
    int nxt, to, v;
}rin[Excalibur<<3];//是凛哦
int Lancer[Excalibur], Fate = 1, Archar;//emmm,head和tot以及sum
int x[5] = {0, -1, 0, 1, 0}, y[5] = {0, 0, -1, 0, 1};

inline void add(int from, int to, int v) {
    rin[++Fate].v = v;
    rin[Fate].to = to;
    rin[Fate].nxt = Lancer[from];
    Lancer[from] = Fate;
}

inline bool bfs(int s, int t) {
    queue<int> Rider;//q
    for(R int i = 0;i <= T;++ i) cur[i] = Lancer[i], dep[i] = 0;
    Rider.push(s); dep[s] = 1;
    while(!Rider.empty()) {
        R int vi = Rider.front();
        Rider.pop();
        for(R int i = Lancer[vi]; i ;i = rin[i].nxt) {
            R int vc = rin[i].to;
            if(!dep[vc] && rin[i].v) {
                dep[vc] = dep[vi] + 1;
                Rider.push(vc);
            }
        }
    }
    return dep[t];
}

int dfs(int s, int t, int flow) {
    if(!flow || s == t) return flow;
    int Caster = 0, Assassin;//增量
    for(R int i = cur[s]; i ;i = rin[i].nxt) {
        R int vc = rin[i].to;
        cur[s] = i;
        if(dep[vc] == dep[s] + 1 && rin[i].v) {
            Assassin = dfs(vc, t, min(flow, rin[i].v));
            if(!Assassin) continue;
            Caster += Assassin; flow -= Assassin;
            rin[i].v -= Assassin; rin[i ^ 1].v += Assassin;
            if(!flow) return Caster;
        }
    }
    return Caster;
}

int Dinic() {
    int res = 0;
    while(bfs(S, T)) res += dfs(S, T, inf);
    return Archar - res;
}

int main() {
    scanf("%d%d",&n, &m);
    //以下是建图,中二到此结束
    S = 0, T = n * m + 1;
    for(R int i = 1;i <= n;++ i)
        for(R int j = 1;j <= m;++ j) {
            R int v; scanf("%d",&v);
            Archar += v;
            R int p = (i - 1) * m + j;
            if((i + j) & 1) { add(S, p, v);add(p, S, 0); }
            else { add(p, T, v);add(T, p, 0); }
        }
    for(R int i = 1;i <= n;++ i) for(R int j = 1;j <= m;++ j)
        if((i + j) & 1)
            for(R int k = 1;k <= 4;++ k) {
                R int f = i + x[k], g = j + y[k];
                if(f < 1 || f > n || g < 1 || g > m) continue;
                R int u = (i - 1) * m + j, v = (f - 1) * m + g;
                add(u, v, inf); add(v, u, 0);
            }
    printf("%d",Dinic());
    return 0;
}

 

posted @ 2019-09-27 20:36  yelir  阅读(335)  评论(0编辑  收藏  举报