agc011_e 题解

一种 luogu 题解区没有的新做法。

考虑一个非常有道理的贪心:假设当前的数是 \(n\),每次选择 \(\le n\) 的最大的递增数 \(v\) ,将 \(n\) 变为 \(n-v\)。

考虑上面的加粗部分比较抽象,实际上,比如样例 \(n=20170312\),那么 \(v=19999999\)。

感性理解就是一定会让最高位没掉。

但是直接做是 \(\mathcal{O}(n^2)\) 的,考虑优化,如果使用线段树来维护高精度可以做到 \(\mathcal{O}(n\log n)\)。

考虑继续优化,发现瓶颈在于每次减去形如 \(\overline{a}99\dots9\),则相当于先加减去一个 \(\overline{a}\times(10^k+1)\),再加 \(1\)。发现前者会刚好消去当前这个数前若干位,而后者均摊是 \(\mathcal{O}(n)\) 的,因此可以优化到 \(\mathcal{O}(n)\)。

代码:

cpp 复制代码
vector<int> num;
string str; cin >> str;
for (char c : str) num.pb(c - '0');
int step = 0, pos = -1, cur = 0;
while (true) {
  ++step;
  if (pos < cur) {
    for (int i = cur; i + 1 < num.size(); ++i) if (num[i] > num[i + 1]) {
      pos = i;
      break;
    }
    if (pos < cur) break;
  }
  int sep = num[pos];
  while (true) {
    int f = num[cur];
    ++cur;
    if (f == sep) break;
  }
  [&]{
    for (int i = (int)num.size() - 1; i >= cur; --i) {
      if (num[i] < 9) {
        ++num[i];
        if (i <= pos + 1) pos = cur - 1;
        return;
      }
      num[i] = 0;
    }
    --cur;
    num[cur] = 1;
    pos = cur - 1;
  }();
  while (num[cur] == 0) ++cur;
}
cout << step << "\n";
相关推荐
dragoooon342 分钟前
[hot100 NO.62~67]
算法
你撅嘴真丑6 分钟前
求矩阵的两对角线上的元素之和 与 sizeof的大作用
线性代数·算法·矩阵
程序员三明治10 分钟前
【面试手撕】如何构造二叉树输入用例?ACM模式,路径总和2解题思路
算法·leetcode·面试·acm·构造二叉树·路径总和
干前端22 分钟前
Message组件和Vue3 进阶:手动挂载组件与 Diff 算法深度解析
javascript·vue.js·算法
ゞ 正在缓冲99%…24 分钟前
2025.12.17华为软开
java·算法
子午30 分钟前
【2026原创】文本情感识别系统~Python+深度学习+textCNN算法+舆情文本+模型训练
python·深度学习·算法
Flash.kkl31 分钟前
递归、搜索与回溯算法概要
数据结构·算法
s090713631 分钟前
【MATLAB】多子阵合成孔径声纳(SAS)成像仿真——基于时域反向投影(BP)算法
算法·matlab·bp算法·合成孔径
Xの哲學32 分钟前
Linux Workqueue 深度剖析: 从设计哲学到实战应用
linux·服务器·网络·算法·边缘计算
sin_hielo37 分钟前
leetcode 3047
数据结构·算法·leetcode