[TJOI2009]战争游戏 - 网络流,最小割

Description

小R正在玩一个战争游戏。游戏地图是一个M行N列的矩阵,每个格子可能是障碍物,也可能是空地,在游戏开始时有若干支敌军分散在不同的空地格子中。每支敌军都可以从当前所在的格子移动到四个相邻的格子之一,但是不能移动到包含障碍物的格子。如果敌军移动出了地图的边界,那么战争就失败了。

现在你的任务是,在敌军开始移动前,通过飞机轰炸使得某些原本是空地的格子变得不可通行,这样就有可能阻止敌军移出地图边界(出于某种特殊的考虑,你不能直接轰炸敌军所在的格子)。由于地形不同的原因,把每个空地格子轰炸成不可通行所需的Bomb数目可能是不同的,你需要计算出要阻止敌军所需的最少的Bomb数。

Input & Output

Input

输入文件的第一行包含两个数M和N,分别表示矩阵的长和宽。接下来M行,每行包含用空格隔开的N个数字,每个数字表示一个格子的情况:若数字为-1,表示这个格子是障碍物;若数字为0,表示这个格子里有一支敌军;若数字为一个正数x,表示这个格子是空地,且把它轰炸成不可通行所需的Bomb数为x。

Output

输出一个数字,表示所需的最少Bomb数。数据保证有解存在。

Sample

Input

4 3
1 2 1
1 10 1
1 0 -1
1 1 1

Output

6

Solution

把每个可轰炸的点拆点,由i向i + ?连一条容量为点权的边,特别地,对于敌军所在的点,由源点向敌军,敌军向拆点各连一条INF,然后在所有非障碍的四连通格子间连双向INF,具体的操作是x -> y + ?, y -> x + ?,最后是所有非障碍边界向汇点连一条INF,这里说的边都包括了对应的反向弧。
求最小割即可。
Code:

#include <iostream>
#include <cstdio>
#include <algorithm>
#include <cstring>
#include <queue>

using std::memset;
using std::min;
using std::max;
using std::swap;
using std::cin;
using std::cout;
using std::endl;
using std::ios;
using std::queue;

const int maxn = 64;
const int INF = 0x3f3f3f3f;

struct edge
{
    int to,nxt,v;
}e[10005];
int n,m,s,t,lnk[10005],level[10005],ptr;
int mtr[32][32],num[32][32];

void add(int bgn,int end,int flow)
{
    e[ptr].v = flow; e[ptr].to = end; e[ptr].nxt = lnk[bgn]; lnk[bgn] = ptr; ++ptr;
    e[ptr].v = 0; e[ptr].to = bgn; e[ptr].nxt = lnk[end]; lnk[end] = ptr; ++ptr;
}
bool bfs()
{
    queue<int> q;
    memset(level,0,sizeof(level));
    level[s]=1;
    q.push(s);
    while(!q.empty()){
        int u=q.front();q.pop();
        for(int p=lnk[u]; ~p; p=e[p].nxt){
            int y=e[p].to;
            if(e[p].v > 0 && !level[y]){
                level[y]=level[u]+1;
                q.push(y);
            }
        }
    }
    if(level[t]==0) return false;
    else return true;
}
int dfs(int x,int fl)
{
    if(x==t)return fl;
    for(int p=lnk[x]; ~p; p=e[p].nxt){
        int y=e[p].to;
        if(level[y]==level[x]+1&&e[p].v!=0){
            int delta=dfs(y,min(fl,e[p].v));
            if(delta > 0){
                e[p].v-=delta;
                e[p^1].v+=delta;
                return delta;
            }
        }
    }
    return 0;
}
int dinic()
{
    int ret=0;
    while(bfs()){
        while(int delta=dfs(s,INF))
            ret+=delta;
    }
    return ret;
}

int main()
{
    ios::sync_with_stdio(false);
    memset(lnk,-1,sizeof(lnk));
    cin >> n >> m;
    t = n*m; s = 0;
    for(int i = 1; i <= n; ++i)
        for(int j = 1; j <= m; ++j)
            cin >> mtr[i][j];
    for(int i = 1; i <= n; ++i)
        for(int j = 1; j <= m; ++j)
        {
            num[i][j] = i * m + j;
            if(mtr[i][j])add(num[i][j],num[i][j] + t,mtr[i][j]);
        }
    for(int i = 1; i <= n; ++i)
        for(int j = 1; j <= m; ++j)
        {
            if(mtr[i][j] == 0) add(s,num[i][j],INF) , add(num[i][j], num[i][j] + t, INF);
            if(i > 1 && mtr[i-1][j] >= 0 && mtr[i][j] >= 0)
                add(num[i][j] + t, num[i-1][j], INF),
                add(num[i-1][j] + t, num[i][j], INF);
            if(j > 1 && mtr[i][j-1] >= 0 && mtr[i][j] >= 0)
                add(num[i][j] + t, num[i][j-1], INF),
                add(num[i][j-1] + t, num[i][j], INF);
            if(i == 1 || i == n || j == 1 || j == m)
                add(num[i][j] + t, (t << 1) + 1, INF);
        }
    t = (t << 1) + 1;
    cout << dinic() << endl;
    return 0;
}

辣鸡博客园居然还有Bomb这种敏感词

原文地址:https://www.cnblogs.com/nishikino-curtis/p/9047576.html

时间: 2024-10-03 22:26:14

[TJOI2009]战争游戏 - 网络流,最小割的相关文章

【bzoj3144】[Hnoi2013]切糕 网络流最小割

题目描述 输入 第一行是三个正整数P,Q,R,表示切糕的长P. 宽Q.高R.第二行有一个非负整数D,表示光滑性要求.接下来是R个P行Q列的矩阵,第z个 矩阵的第x行第y列是v(x,y,z) (1≤x≤P, 1≤y≤Q, 1≤z≤R). 100%的数据满足P,Q,R≤40,0≤D≤R,且给出的所有的不和谐值不超过1000. 输出 仅包含一个整数,表示在合法基础上最小的总不和谐值. 样例输入 2 2 2 1 6 1 6 1 2 6 2 6 样例输出 6 题目大意 给定一个p行q列的矩阵,每个位置可以

二分图&amp;网络流&amp;最小割等问题的总结

二分图基础: 最大匹配:匈牙利算法 最小点覆盖=最大匹配 最小边覆盖=总节点数-最大匹配 最大独立集=点数-最大匹配 网络流: 带下界网络流 最小割问题的总结: *意义 1.加inf的边表示不能被割,通常用于体现某个点必须属于某个集合 连边(s,u,w)代表如果u不在s割的话需要付出代价w 2.连边(u,v,w)代表如果u在s割,v在t割需要付出代价w 但注意,如果u在t割,v在s割是不需要付出代价的. 那么如果连边(u,v,w)以及(v,u,w)则说明当u与v所属割不同的时候需要付出代价w *

HDU 2435 There is a war (网络流-最小割)

There is a war Problem Description There is a sea. There are N islands in the sea. There are some directional bridges connecting these islands. There is a country called Country One located in Island 1. There is another country called Country Another

【bzoj3630】[JLOI2014]镜面通道 对偶图+计算几何+网络流最小割

题目描述 在一个二维平面上,有一个镜面通道,由镜面AC,BD组成,AC,BD长度相等,且都平行于x轴,B位于(0,0).通道中有n个外表面为镜面的光学元件,光学元件α为圆形,光学元件β为矩形(这些元件可以与其他元件和通道有交集,具体看下图).光线可以在AB上任一点以任意角度射入通道,光线不会发生削弱.当出现元件与元件,元件和通道刚好接触的情况视为光线无法透过(比如两圆相切).现在给出通道中所有元件的信息(α元件包括圆心坐标和半径xi,yi,ri,β元件包括左下角和右上角坐标x1,y1,x2,y2

【bzoj2127】happiness 网络流最小割

题目描述 高一一班的座位表是个n*m的矩阵,经过一个学期的相处,每个同学和前后左右相邻的同学互相成为了好朋友.这学期要分文理科了,每个同学对于选择文科与理科有着自己的喜悦值,而一对好朋友如果能同时选文科或者理科,那么他们又将收获一些喜悦值.作为计算机竞赛教练的scp大老板,想知道如何分配可以使得全班的喜悦值总和最大. 输入 第一行两个正整数n,m.接下来是六个矩阵第一个矩阵为n行m列 此矩阵的第i行第j列的数字表示座位在第i行第j列的同学选择文科获得的喜悦值.第二个矩阵为n行m列 此矩阵的第i行

【bzoj2132】圈地计划 网络流最小割

题目描述 最近房地产商GDOI(Group of Dumbbells Or Idiots)从NOI(Nuts Old Idiots)手中得到了一块开发土地.据了解,这块土地是一块矩形的区域,可以纵横划分为N×M块小区域.GDOI要求将这些区域分为商业区和工业区来开发.根据不同的地形环境,每块小区域建造商业区和工业区能取得不同的经济价值.更具体点,对于第i行第j列的区域,建造商业区将得到Aij收益,建造工业区将得到Bij收益.另外不同的区域连在一起可以得到额外的收益,即如果区域(I,j)相邻(相邻

ZOJ3792_Romantic Value(网络流/最小割=最大流/找割边)

解题报告 题目传送门 题意: 给出一个无向图,以及起点与终点.要删除一些边使得起点与终点不连通,在删掉边的权值之和最小的情况下要求删除的边数尽量少. 求出一个比值:剩余边数权值和/删除的边数. 思路: 明显的让起点终点达不到就是一个最小割,用最大流可以求出. 但是求割边边数就不会了,没做过最小割的求割边问题. 割边一定是残留网络中零流的边,但零流不一定是割边. 飞神的想法很奇特.链接传送 可以把残留网络的零流的边设成容量为1,其他设成无穷,再求一次最大流.最后流量一定等于割边边数 另外: 还有一

HDU 4289 Control (网络流-最小割)

Control Problem Description You, the head of Department of Security, recently received a top-secret information that a group of terrorists is planning to transport some WMD 1 from one city (the source) to another one (the destination). You know their

POJ3469_Dual Core CPU(网络流/最小割=最大流/模版)----Dinic模版2.0

解题报告 题目传送门 题意: 双核CPU,n个模块,每个模块必须运行在某个CPU核心上,每个模块在cpu单核的消耗A和B,M对模块要共享数据,如果在同一个核心上不用消耗,否则需要耗费.安排N个模块,使得总耗费最小 思路: 将两个cpu核心看成源点和汇点,其他模块分别与源点汇点连线(表示每个模块可以在任意cpu上运行),m对模块分别连双向边,要使得模块只能在一个cpu上运行,就是找到一个割,源点和汇点必不联通,耗费最少就是最小割,最小割最大流原理转换成求最大流. 这题数据大,没优化TLE了,加了两