题目链接:BZOJ - 3144

题目分析

题意:在 P * Q 的方格上填数字,可以填 [1, R] 。

在 (x, y) 上填 z 会有 V[x][y][z] 的代价。限制:相邻两个格子填的数字的差的绝对值不能超过 D 。

求一个合法的最小总代价。

这道题是一个最小割模型,直接说建图吧。

建图:每个点 (x, y) 拆成 R 个点,(x, y, z) 代表 (x, y) 填 z。

然后从 S 向 (*, *, 1) 连 INF ,从 (*, *, R) 向 T 连 INF 。

然后对于 (i, j, k) ,向 (i, j, k + 1) 连 V[i][j][k] 的边。

对于与 (i, j) 相邻的 (i', j') ,从 (i, j, k) 向 (i', j', k - D) 连INF。

这样,割掉 (i, j, k) -> (i, j, k + 1) 的边,就是在 (i, j) 填了 k。

如果 (i, j) 填了 k ,与 (i, j) 相邻的 (i', j') 就只能填比 k - D 大的数字。

代码

#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <cmath>
#include <algorithm> using namespace std; const int MaxN = 40, MaxNode = 65000 + 15, INF = 999999999, Dx[5] = {0, 0, 1, -1}, Dy[5] = {1, -1, 0, 0}; int P, Q, R, D, S, T, Tot, MaxFlow;
int Idx[MaxN][MaxN][MaxN], V[MaxN][MaxN][MaxN], Num[MaxNode], d[MaxNode]; struct Edge
{
int v, w;
Edge *Next, *Other;
} E[MaxNode * 6], *Pi = E, *Point[MaxNode], *Last[MaxNode]; inline void AddEdge(int x, int y, int z)
{
Edge *Q = ++Pi; ++Pi;
Pi -> v = y; Pi -> w = z;
Pi -> Next = Point[x]; Point[x] = Pi; Pi -> Other = Q;
Q -> v = x; Q -> w = 0;
Q -> Next = Point[y]; Point[y] = Q; Q -> Other = Pi;
} inline int gmin(int a, int b) {return a < b ? a : b;} int DFS(int Now, int Flow)
{
if (Now == T) return Flow;
int ret = 0;
for (Edge *j = Last[Now]; j; j = j -> Next)
if (j -> w && d[Now] == d[j -> v] + 1)
{
Last[Now] = j;
int p = DFS(j -> v, gmin(j -> w, Flow - ret));
ret += p; j -> w -= p; j -> Other -> w += p;
if (ret == Flow) return ret;
}
if (d[S] >= Tot) return ret;
if (--Num[d[Now]] == 0) d[S] = Tot;
++Num[++d[Now]];
Last[Now] = Point[Now];
return ret;
} inline bool Inside(int x, int y)
{
if (x < 1 || x > P) return false;
if (y < 1 || y > Q) return false;
return true;
} int main()
{
scanf("%d%d%d%d", &P, &Q, &R, &D);
for (int i = 1; i <= R; ++i)
for (int j = 1; j <= P; ++j)
for (int k = 1; k <= Q; ++k)
scanf("%d", &V[j][k][i]);
for (int i = 1; i <= P; ++i)
for (int j = 1; j <= Q; ++j)
for (int k = 1; k <= R; ++k)
Idx[i][j][k] = (k - 1) * (P * Q) + (i - 1) * Q + j;
Tot = Idx[P][Q][R]; S = ++Tot; T = ++Tot;
for (int i = 1; i <= P; ++i)
for (int j = 1; j <= Q; ++j)
{
Idx[i][j][1] = S;
Idx[i][j][R + 1] = T;
for (int k = 1; k <= R; ++k)
{
AddEdge(Idx[i][j][k], Idx[i][j][k + 1], V[i][j][k]);
if (k > D + 1)
{
int x, y;
for (int f = 0; f < 4; ++f)
{
x = i + Dx[f]; y = j + Dy[f];
if (!Inside(x, y)) continue;
AddEdge(Idx[i][j][k], Idx[x][y][k - D], INF);
}
}
}
}
MaxFlow = 0;
memset(d, 0, sizeof(d));
memset(Num, 0, sizeof(Num)); Num[0] = Tot;
for (int i = 1; i <= Tot; ++i) Last[i] = Point[i];
while (d[S] < Tot) MaxFlow += DFS(S, INF);
printf("%d\n", MaxFlow);
return 0;
}

  

[BZOJ 3144] [Hnoi2013] 切糕 【最小割】的更多相关文章

  1. bzoj 3144: [Hnoi2013]切糕 最小割

    3144: [Hnoi2013]切糕 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 681  Solved: 375[Submit][Status] ...

  2. bzoj 3144 [Hnoi2013]切糕——最小割

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3144 一根纵轴上切一个点,可以把一根纵轴上的点连成一串来体现.自己的写法是每个点连向前一个点 ...

  3. bzoj3144 [HNOI2013]切糕(最小割)

    bzoj3144 [HNOI2013]切糕(最小割) bzoj Luogu 题面描述见上 题解时间 一开始我真就把这玩意所说的切面当成了平面来做的 事实上只是说相邻的切点高度差都不超过 $ d $ 对 ...

  4. BZOJ 3144: [Hnoi2013]切糕

    3144: [Hnoi2013]切糕 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1495  Solved: 819[Submit][Status] ...

  5. bzoj 3144 [Hnoi2013]切糕【最小割+dinic】

    都说了是'切'糕所以是最小割咯 建图: 每个点向下一层连容量为这个点的val的边,S向第一层连容量为inf的边,最后一层向T连容量为自身val的边,即割断这条边相当于\( f(i,j) \)选择了当前 ...

  6. 【BZOJ3144】[Hnoi2013]切糕 最小割

    [BZOJ3144][Hnoi2013]切糕 Description Input 第一行是三个正整数P,Q,R,表示切糕的长P. 宽Q.高R.第二行有一个非负整数D,表示光滑性要求.接下来是R个P行Q ...

  7. BZOJ3144[Hnoi2013]切糕——最小割

    题目描述 输入 第一行是三个正整数P,Q,R,表示切糕的长P. 宽Q.高R.第二行有一个非负整数D,表示光滑性要求.接下来是R个P行Q列的矩阵,第z个 矩阵的第x行第y列是v(x,y,z) (1≤x≤ ...

  8. 【刷题】BZOJ 3144 [Hnoi2013]切糕

    Description Input 第一行是三个正整数P,Q,R,表示切糕的长P. 宽Q.高R.第二行有一个非负整数D,表示光滑性要求.接下来是R个P行Q列的矩阵,第z个 矩阵的第x行第y列是v(x, ...

  9. Luogu P3227 [HNOI2013]切糕 最小割

    首先推荐一个写的很好的题解,个人水平有限只能写流水账,还请见谅. 经典的最小割模型,很多人都说这个题是水题,但我还是被卡了=_= 技巧:加边表示限制 在没有距离\(<=d\)的限制时候,我们对每 ...

随机推荐

  1. shell 获取当前ip

    HOST_IP=$(hostname --all-ip-addresses | awk '{print $1}')

  2. mybatis04 根据用户名称模糊查询用户信息

    根据用户名称模糊查询用户信息可能返回多条记录. 1.1.1User.xml 编码 如果用%进行模糊查询,#{}表示一个占位符会被翻译为一个?号(SELECT * FROM USER WHERE id= ...

  3. MYSQL参数学习---------------- 张碧池

    http://pottievil.com/category/mysql/mysql%E5%8F%82%E6%95%B0/

  4. 如何查看电脑的链接WIFI密码

    这个查看起来确实有难度,一般软件也难以破解.不如你试试在浏览器输入192.168.1.1,账户密码都是admin然后进入路由设置里.无线安全设置里有.

  5. .net mvc 发布部署到机器上

    这样会自动在C:\Inetpub\wwwroot 会出现这个文件夹. 上述步骤都是在安装了 VS2010.MVC3.0和的电脑上操作的 接下来进行服务器的部署 :1. 安装Microsoft .net ...

  6. Android(java)学习笔记178:BroadcastReceiver之 自定义广播

    广播使用:               电台:对外发送信号.---------电台发送广播(可以自定义)               收音机:接收电台的信号.-----广播接收者 这里,我们就说明自定 ...

  7. Chapter 5. The Gradle Wrapper 关于gradle wrapper

    Most tools require installation on your computer before you can use them. If the installation is eas ...

  8. CentOS Linux修改系统时区

    ln –sf /usr/share/zoneinfo/Hongkong /etc/localtime

  9. Mybatis下配置调用Oracle自定义函数返回的游标结果集

    在ibatis和Mybatis对存储过程和函数函数的调用的配置Xml是不一样的,以下是针对Mybatis 3.2的环境进行操作的. 第一步配置Mapper的xml内容 <mapper names ...

  10. C#当中的多线程_线程同步

    第2章 线程同步 原来以为线程同步就是lock,monitor等呢,看了第二章真是大开眼界啊! 第一章中我们遇到了一个叫做竞争条件的问题.引起的原因是没有进行正确的线程同步.当一个线程在执行操作时候, ...