悬线法
引入
悬线法的适用范围是单调栈的子集。具体来说,悬线法可以应用于满足以下条件的题目:
- 需要在扫描序列时维护单调的信息;
- 可以使用单调栈解决;
- 不需要在单调栈上二分。
看起来悬线法可以被替代,用处不大,但是悬线法概念比单调栈简单,更适合初学 OI 的选手理解并解决最大子矩阵等问题。
例题
SPOJ HISTOGRA - Largest Rectangle in a Histogram
 大意:在一条水平线上有 𝑛 个宽为 1
 个宽为 1 的矩形,求包含于这些矩形的最大子矩形面积。
 的矩形,求包含于这些矩形的最大子矩形面积。
悬线,就是一条竖线,这条竖线有初始位置和高度两个性质,可以在其上端点不超过当前位置的矩形高度的情况下左右移动。
对于一条悬线,我们在这条上端点不超过当前位置的矩形高度且不移出边界的前提下,将这条悬线左右移动,求出其最多能向左和向右扩展到何处,此时这条悬线扫过的面积就是包含这条悬线的尽可能大的矩形。容易发现,最大子矩形必定是包含一条初始位置为 𝑖 ,高度为 ℎ𝑖
,高度为 ℎ𝑖 的悬线。枚举实现这个过程的时间复杂度为 𝑂(𝑛2)
 的悬线。枚举实现这个过程的时间复杂度为 𝑂(𝑛2) ,但是我们可以用悬线法将其优化到 𝑂(𝑛)
,但是我们可以用悬线法将其优化到 𝑂(𝑛) 。
。
我们考虑如何快速找到悬线可以到达的最左边的位置。
过程
定义 𝑙𝑖 为当前找到的 𝑖
 为当前找到的 𝑖 位置的悬线能扩展到的最左边的位置,容易得到 𝑙𝑖
 位置的悬线能扩展到的最左边的位置,容易得到 𝑙𝑖 初始为 𝑖
 初始为 𝑖 ,我们需要进一步判断还能不能进一步往左扩展。
,我们需要进一步判断还能不能进一步往左扩展。
- 如果当前 𝑙𝑖 =1 ,则已经扩展到了边界,不可以。 ,则已经扩展到了边界,不可以。
- 如果当前 𝑎𝑖 >𝑎𝑙𝑖−1 ,则从当前悬线扩展到的位置不能再往左扩展了。 ,则从当前悬线扩展到的位置不能再往左扩展了。
- 如果当前 𝑎𝑖 ≤𝑎𝑙𝑖−1 ,则从当前悬线还可以往左扩展,并且 𝑙𝑖 −1 ,则从当前悬线还可以往左扩展,并且 𝑙𝑖 −1 位置的悬线能向左扩展到的位置,𝑖 位置的悬线能向左扩展到的位置,𝑖 位置的悬线一定也可以扩展到,于是我们将 𝑙𝑖 位置的悬线一定也可以扩展到,于是我们将 𝑙𝑖 更新为 𝑙𝑙𝑖−1 更新为 𝑙𝑙𝑖−1 ,并继续执行判断。 ,并继续执行判断。
通过摊还分析,可以证明每个 𝑙𝑖 最多会被其他的 𝑙𝑗
 最多会被其他的 𝑙𝑗 遍历到一次,因此时间复杂度为 𝑂(𝑛)
 遍历到一次,因此时间复杂度为 𝑂(𝑛) 。
。
实现
参考代码
 |  1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25 | #include <algorithm>
#include <iostream>
using std::max;
constexpr int N = 100010;
int n, a[N];
int l[N], r[N];
long long ans;
using std::cin;
using std::cout;
int main() {
  cin.tie(nullptr)->sync_with_stdio(false);
  while (cin >> n, n) {
    ans = 0;
    for (int i = 1; i <= n; i++) cin >> a[i], l[i] = r[i] = i;
    for (int i = 1; i <= n; i++)
      while (l[i] > 1 && a[i] <= a[l[i] - 1]) l[i] = l[l[i] - 1];
    for (int i = n; i >= 1; i--)
      while (r[i] < n && a[i] <= a[r[i] + 1]) r[i] = r[r[i] + 1];
    for (int i = 1; i <= n; i++)
      ans = max(ans, (long long)(r[i] - l[i] + 1) * a[i]);
    cout << ans << '\n';
  }
  return 0;
}
 | 
UVa1619 感觉不错 Feel Good
 对于一个长度为 𝑛 的数列,找出一个子区间,使子区间内的最小值与子区间内元素和的乘积最大,要求在满足舒适值最大的情况下最小化长度,最小化长度的情况下最小化左端点序号。
 的数列,找出一个子区间,使子区间内的最小值与子区间内元素和的乘积最大,要求在满足舒适值最大的情况下最小化长度,最小化长度的情况下最小化左端点序号。
本题中我们可以考虑枚举最小值,将每个位置的数 𝑎𝑖 当作最小值,并考虑从 𝑖
 当作最小值,并考虑从 𝑖 向左右扩展,找到满足 𝑟min𝑗=𝑙𝑎𝑗 =𝑎𝑖
 向左右扩展,找到满足 𝑟min𝑗=𝑙𝑎𝑗 =𝑎𝑖 的尽可能向左右扩展的区间 [𝑙,𝑟]
 的尽可能向左右扩展的区间 [𝑙,𝑟]![[l, r]]() 。这样本题就被转化成了悬线法模型。
。这样本题就被转化成了悬线法模型。
参考代码
 |  1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39 | #include <cstring>
#include <iostream>
constexpr int N = 100010;
int n, a[N], l[N], r[N];
long long sum[N];
long long ans;
int ansl, ansr;
bool fir = true;
using std::cin;
using std::cout;
int main() {
  cin.tie(nullptr)->sync_with_stdio(false);
  while (cin >> n) {
    memset(a, -1, sizeof(a));
    if (!fir)
      cout << '\n';
    else
      fir = false;
    ans = 0;
    ansl = ansr = 1;
    for (int i = 1; i <= n; i++) {
      cin >> a[i];
      sum[i] = sum[i - 1] + a[i];
      l[i] = r[i] = i;
    }
    for (int i = 1; i <= n; i++)
      while (a[l[i] - 1] >= a[i]) l[i] = l[l[i] - 1];
    for (int i = n; i >= 1; i--)
      while (a[r[i] + 1] >= a[i]) r[i] = r[r[i] + 1];
    for (int i = 1; i <= n; i++) {
      long long x = a[i] * (sum[r[i]] - sum[l[i] - 1]);
      if (ans < x || (ans == x && ansr - ansl > r[i] - l[i]))
        ans = x, ansl = l[i], ansr = r[i];
    }
    cout << ans << '\n' << ansl << ' ' << ansr << '\n';
  }
  return 0;
}
 | 
最大子矩形
P4147 玉蟾宫
 给定一个 𝑛 ×𝑚 的包含
 的包含 'F' 和 'R' 的矩阵,求其面积最大的子矩阵的面积 ×3 ,使得这个子矩阵中的每一位的值都为
,使得这个子矩阵中的每一位的值都为 'F'。
我们会发现本题的模型和第一题的模型很像。仔细分析,发现如果我们每次只考虑某一行的所有元素,将位置 (𝑥,𝑦) 的元素尽可能向上扩展的距离作为该位置的悬线长度,那最大子矩阵一定是这些悬线向左右扩展得到的尽可能大的矩形中的一个。
 的元素尽可能向上扩展的距离作为该位置的悬线长度,那最大子矩阵一定是这些悬线向左右扩展得到的尽可能大的矩形中的一个。
参考代码
 |  1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30 | #include <algorithm>
#include <iostream>
int m, n, a[1010], l[1010], r[1010], ans;
using std::cin;
using std::cout;
int main() {
  cin.tie(nullptr)->sync_with_stdio(false);
  cin >> n >> m;
  for (int i = 1; i <= n; i++) {
    for (int j = 1; j <= m; j++) {
      l[j] = r[j] = j;
    }
    char s[3];
    for (int j = 1; j <= m; j++) {
      cin >> s;
      if (s[0] == 'F')
        a[j]++;
      else if (s[0] == 'R')
        a[j] = 0;
    }
    for (int j = 1; j <= m; j++)
      while (l[j] != 1 && a[l[j] - 1] >= a[j]) l[j] = l[l[j] - 1];
    for (int j = m; j >= 1; j--)
      while (r[j] != m && a[r[j] + 1] >= a[j]) r[j] = r[r[j] + 1];
    for (int j = 1; j <= m; j++) ans = std::max(ans, (r[j] - l[j] + 1) * a[j]);
  }
  cout << ans * 3;
  return 0;
}
 | 
习题
本页面最近更新:2025/10/13 16:54:57,更新历史
发现错误?想一起完善? 在 GitHub 上编辑此页!
本页面贡献者:countercurrent-time, StudyingFather, GekkaSaori, H-J-Granger, NachtgeistW, Enter-tainer, Ir1d, sshwy, AngelKitty, CCXXXI, cjsoft, diauweb, Early0v0, ezoixx130, Konano, LovelyBuggies, Makkiy, mgt, minghu6, P-Y-Y, PotassiumWings, SamZhangQingChuan, Suyun514, weiyong1024, Ahacad, c-forrest, dis-GU-ise, GavinZhengOI, Gesrua, gi-b716, Henry-ZHR, HeRaNO, hsfzLZH1, iamtwz, kenlig, kxccc, lychees, mwsht, ouuan, Peanut-Tang, SukkaW, Tiphereth-A
本页面的全部内容在 CC BY-SA 4.0 和 SATA 协议之条款下提供,附加条款亦可能应用