Description

给定一个由 \(n\) 行数字组成的数字梯形如下图所示。

梯形的第一行有 \(m\) 个数字。从梯形的顶部的 \(m\) 个数字开始,在每个数字处可以沿左下或右下方向移动,形成一条从梯形的顶至底的路径。

分别遵守以下规则:

  1. 从梯形的顶至底的 \(m\) 条路径互不相交;
  2. 从梯形的顶至底的 \(m\) 条路径仅在数字结点处相交;
  3. 从梯形的顶至底的 \(m\) 条路径允许在数字结点相交或边相交。

Limitation

\(1~\leq~n,~m~\leq~20\)

Solution

解释一下题意,边不相交指的是不能有两条路径同时经过 \(u \rightarrow~v\) 的路径。

先考虑限制 \(3\),也就是没有限制的情况,做法非常显然,上一层向下一层的数字连边,容量为无穷代表这条边可以走无穷次,花费为 \(0\)。每个数字都拆一下点,两个点之间连边容量为无穷,代表可以选这个点无数次,花费为这个点的权值代表经过他付出的代价,\(s\) 向第一层连容量为 \(1\) 费用为 \(0\) 的边,最后一层向 \(t\) 连容量为无穷费用为 \(0\) 的边,跑最大费用最大流即可。

考虑限制 \(2\),一条边只能经过一次,于是将边的容量置为 \(1\) 即可。

考虑限制 \(1\),同理将点的容量置成 \(1\) 即可。

然后如果你Wa前两个点需要注意梯形的最下面会有 \(n + m\) 个点而不是 \(m\) 个

Code

#include <cstdio>
#include <cstring>
#include <queue>
#include <algorithm>
#ifdef ONLINE_JUDGE
#define freopen(a, b, c)
#endif typedef long long int ll; namespace IPT {
const int L = 1000000;
char buf[L], *front=buf, *end=buf;
char GetChar() {
if (front == end) {
end = buf + fread(front = buf, 1, L, stdin);
if (front == end) return -1;
}
return *(front++);
}
} template <typename T>
inline void qr(T &x) {
char ch = IPT::GetChar(), lst = ' ';
while ((ch > '9') || (ch < '0')) lst = ch, ch=IPT::GetChar();
while ((ch >= '0') && (ch <= '9')) x = (x << 1) + (x << 3) + (ch ^ 48), ch = IPT::GetChar();
if (lst == '-') x = -x;
} namespace OPT {
char buf[120];
} template <typename T>
inline void qw(T x, const char aft, const bool pt) {
if (x < 0) {x = -x, putchar('-');}
int top=0;
do {OPT::buf[++top] = static_cast<char>(x % 10 + '0');} while (x /= 10);
while (top) putchar(OPT::buf[top--]);
if (pt) putchar(aft);
} const int maxn = 5010;
const int INF = 0x3f3f3f3f; struct Edge {
int u, v, flow, fee;
Edge *nxt, *bk;
Edge(const int _u, const int _v, const int _flow, const int _fee, Edge* &h)
: u(_u), v(_v), flow(_flow), fee(_fee), nxt(h) {
h = this;
} ~Edge() {
if (this->nxt) delete this->nxt;
}
};
Edge *hd[maxn], *pre[maxn];
inline void cont(const int _u, const int _v, const int _flow, const int _fee) {
auto u = new Edge(_u, _v, _flow, _fee, hd[_u]), v = new Edge(_v, _u, 0, -_fee, hd[_v]);
(u->bk = v)->bk = u;
} int n, m, s, t, ans;
int MU[maxn][maxn], id[maxn][maxn][2], dist[maxn], canag[maxn];
bool inq[maxn];
std::queue<int>Q; void EK();
bool spfa();
void argu();
void setedge(int x);
void setpoint(int x); int main() {
freopen("1.in", "r", stdin);
qr(m); qr(n);
for (int i = 1; i <= n; ++i) {
for (int j = 1, k = m + i; j < k; ++j) {
id[i][j][0] = ++t; id[i][j][1] = ++t;
qr(MU[i][j]);
}
}
s = ++t; ++t;
setpoint(1);
setedge(1);
EK();
setpoint(INF);
setedge(1);
EK();
setpoint(INF);
setedge(INF);
EK();
return 0;
} void setpoint(int x) {
for (int i = 1; i <= t; ++i) {
delete hd[i];
hd[i] = NULL;
}
for (int i = 1; i <= m; ++i) {
cont(s, id[1][i][0], 1, 0);
}
for (int i = 1; i <= n; ++i) {
for (int j = 1, k = i + m - 1; j <= k; ++j) {
cont(id[i][j][0], id[i][j][1], x, MU[i][j]);
}
}
} void setedge(int x) {
for (int i = 1; i < n; ++i) {
int di = i + 1;
for (int j = 1, k = i + m - 1; j <= k; ++j) {
cont(id[i][j][1], id[di][j][0], x, 0);
cont(id[i][j][1], id[di][j + 1][0], x, 0);
}
}
for (int j = 1, k = m + n - 1; j <= k; ++j) cont(id[n][j][1], t, INF, 0);
} void EK() {
ans = 0;
while (spfa()) argu();
qw(ans, '\n', true);
} bool spfa() {
memset(canag, 0, sizeof canag);
for (int i = 1; i <= t; ++i) dist[i] = -INF;
dist[s] = 0; Q.push(s); canag[s] = INF;
while (!Q.empty()) {
int u = Q.front(); Q.pop(); inq[u] = false;
for (auto e = hd[u]; e; e = e->nxt) if (e->flow > 0) {
int v = e->v;
if (dist[v] < (dist[u] + e->fee)) {
dist[v] = dist[u] + e->fee;
if (!inq[v]) Q.push(v);
inq[v] = true;
canag[v] = std::min(canag[u], e->flow);
pre[v] = e;
}
}
}
return dist[t] != -INF;
} void argu() {
ans += canag[t] * dist[t];
for (auto e = pre[t]; e; e = pre[e->u]) {
e->flow -= canag[t]; e->bk->flow += canag[t];
}
}

【费用流】【网络流24题】【P4013】 数字梯形问题的更多相关文章

  1. Libre 6010「网络流 24 题」数字梯形 (网络流,最大费用最大流)

    Libre 6010「网络流 24 题」数字梯形 (网络流,最大费用最大流) Description 给定一个由n 行数字组成的数字梯形如下图所示.梯形的第一行有m 个数字.从梯形的顶部的m 个数字开 ...

  2. LOJ #6010. 「网络流 24 题」数字梯形

    #6010. 「网络流 24 题」数字梯形   题目描述 给定一个由 n nn 行数字组成的数字梯形如下图所示.梯形的第一行有 m mm 个数字.从梯形的顶部的 m mm 个数字开始,在每个数字处可以 ...

  3. 2018.10.15 loj#6010. 「网络流 24 题」数字梯形(费用流)

    传送门 费用流经典题. 按照题目要求建边. 为了方便我将所有格子拆点,三种情况下容量分别为111,infinfinf,infinfinf,费用都为validi,jval_{id_{i,j}}valid ...

  4. 【PowerOJ1751&网络流24题】数字梯形问题(费用流)

    题意: 思路: [问题分析] 求图的最大权不相交路径及其变种,用费用最大流解决. [建模方法] 规则(1) 把梯形中每个位置抽象为两个点<i.a>,<i.b>,建立附加源S汇T ...

  5. 【刷题】LOJ 6010 「网络流 24 题」数字梯形

    题目描述 给定一个由 \(n\) 行数字组成的数字梯形如下图所示.梯形的第一行有 \(m\) 个数字.从梯形的顶部的 \(m\) 个数字开始,在每个数字处可以沿左下或右下方向移动,形成一条从梯形的顶至 ...

  6. LG2770/LOJ6122 航空路线问题 费用流 网络流24题

    问题描述 LG2770 LOG6122 题解 教训:关掉流同步之后就不要用其他输入输出方式了. 拆点. 两个拆点之间连\((1,1)\),其他连\((1,0)\) \(\mathrm{Code}\) ...

  7. cogs_14_搭配飞行员_(二分图匹配+最大流,网络流24题#01)

    描述 http://cojs.tk/cogs/problem/problem.php?pid=14 有一些正飞行员和副飞行员,给出每个正飞行员可以和哪些副飞行员一起飞.一架飞机上必须一正一副,求最多多 ...

  8. LG2766 最长不下降子序列问题 最大流 网络流24题

    问题描述 LG2766 题解 \(\mathrm{Subtask 1}\) 一个求最长不下降子序列的问题,发现\(n \le 500\),直接\(O(n^2)\)暴力DP即可. \(\mathrm{S ...

  9. 【网络流24题】最长k可重线段集(费用流)

    [网络流24题]最长k可重线段集(费用流) 题面 Cogs的数据有问题 Loj 洛谷 题解 这道题和最长k可重区间集没有区别 只不过费用额外计算一下 但是,还是有一点要注意的地方 这里可以是一条垂直的 ...

  10. 【网络流24题】最长k可重区间集(费用流)

    [网络流24题]最长k可重区间集(费用流) 题面 Cogs Loj 洛谷 题解 首先注意一下 这道题目里面 在Cogs上直接做就行了 洛谷和Loj上需要判断数据合法,如果\(l>r\)就要交换\ ...

随机推荐

  1. python-五行红旗实现

    import turtle """ 绘制五星红旗 作者:zxj 版本:1.0 """ # 绘制矩形函数 def giant(leg,hig) ...

  2. 基于tensorflow实现mnist手写识别 (多层神经网络)

    标题党其实也不多,一个输入层,三个隐藏层,一个输出层 老样子先上代码 导入mnist的路径很长,现在还记不住 import tensorflow as tf import tensorflow.exa ...

  3. 将本地开发完的SDK代码上传到SVN上面:an error occurred while contacting the repository The server may be unreachable or the URL may be incorrect

    将本地开发完的SDK代码上传到SVN上面:an error occurred while contacting the repository  The server may be unreachabl ...

  4. 【每日scrum】第一次冲刺day5

    请教以前做过类似软件的同学,受益匪浅,启发自己

  5. 项目Beta冲刺(团队)随笔集

    凡事预则立 项目Beta冲刺准备 第一天 项目Beta冲刺(团队)第一天 第二天 项目Beta冲刺(团队)第二天 第三天 项目Beta冲刺(团队)第三天 第四天 项目Beta冲刺(团队)第四天 第五天 ...

  6. golang type

    参考链接 https://blog.csdn.net/tzs919/article/details/53571632 type是golang中非常重要的关键字,常见的就是定义结构体,但是其功能远不止是 ...

  7. fx投影效果分离

    虽然忙着花花二期三期bug.bug  ing,修改等待中突然看见一张logo.文字下面的阴影图,如下,就满脑子在想阴影到底咋做的.. 七拼八凑的尝试后大体样子是有,终究没有上图那种字体轮廓的阴影... ...

  8. webpack命令局部运行的几种方法

    webpack命令局部运行的几种方法   1. 第一种,先全局安装webpack 命令:npm install -g webpack 然后再在项目内安装 命令:npm install webpack ...

  9. hdu 4576 (简单dp+滚动数组)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4576 题意:给出1~n的环,m个操作,每次能顺时针或逆时针走w步,询问最后在l~r这段区间内概率.(1 ...

  10. 关于 WinScp 的一点使用经验

    在嵌入式平台下 是,使用SSH登陆,可以使用WinScp提供的图形界面,支持拖拽,鼠标直接打开,甚是好用. 使用WinScp 登陆的方式有,Scp和sftp两种,之前一只用scp,但后来出现了如下错误 ...