本文共 2188 字,大约阅读时间需要 7 分钟。
一、前言
我在阅读众多博客后,发现关于为什么最底层的宽度最小且高度最高的解释尚不充分,因此决定通过自己的思考得以探索。由于自身缺乏专业视角,仅能提供一个大致的证明思路。
二、证明
(1)LH sb 证明1,暴力讨论法
直接从正面解释高度最值的条件较为困难,因此我们尝试通过柯老师提出的“正难则反”和“better than anyone”思想来探讨问题。
假设当最底层的宽度为f[i]时,干草堆的高度为一个最大值。为了证明当最底层宽度增大时,堆高反而会降低,我们可以进行以下分析:
要将草堆从上层移动到底层,如果底层宽度变大而不新添加草堆,唯一的方法是从上层取出部分草堆将其放入最底层。
(1)上层不满足要求
(2)上层满足要求
(2)证明2,由实现得到的灵感
重新审视问题,以外部视角进行验证:
三、实现
基于上述证明,轻松编写代码:
#include <_vector>#include#include using namespace std;#define LL long long#define ULL unsigned long longtypedef uint64_t ULL;template void read(T &x) { x = 0; ULL f = 1; char c = getchar(); while (c < '0' || c > '9') { if (c == '-') f = -1; c = getchar(); } while (c >= '0' && c <= '9') { x = (x < 1) ? x : x; x = x * 10 + (c - '0'); c = getchar(); } x *= f;}void write(ULL x) { if (x < 0) { x = -x; putchar('-'); } if (x > 9) { write(x / 10); } putchar(x % 10 + '0');}ULL Max(ULL x, ULL y) { return x > y ? x : y; }ULL Min(ULL x, ULL y) { return x < y ? x : y; }ULL Abs(ULL x) { return x > 0 ? x : -x; }const int Maxn = 1e5 + 5;int n;int a[Maxn], f[Maxn], h[Maxn];ULL fall[Maxn];int main() { read(n); for (int i = 1; i <= n; i++) { read(a[i]); } for (int i = n; i >= 1; i--) { fall[i] = (fall[i+1] + a[i]); } for (int i = n; i >= 1; i--) { int j; do { add(j); j = q[tt]; } while (f[j] + fall[j] <= f[tt] + fall[tt]); if (j <= n+1) { f[i] = fall[i] - fall[j]; h[i] = h[j] + 1; } break; } write(h[1]); return 0;}void add(int x) { while (hh <= tt && (f[x] + fall[x]) <= f[q[tt]] + fall[q[tt]]) { tt--; q[hh] = x; hh++; } q[tt] = x; tt++;}
代码逻辑为:
本文完整解释了最底层宽度最小时高度时的最佳状态,方法简单直观,易于理解和复现。
转载地址:http://vfbwk.baihongyu.com/