学习笔记|LeetCode 739 每日温度:从暴力枚举到单调栈线性最优解

学习笔记|LeetCode 739 每日温度:从暴力枚举到单调栈线性最优解

一、问题回顾

给定每日气温数组 temperatures,要求输出等长数组 ans

  • ans[i] 表示第 i 天之后,首次出现更高气温需要等待的天数
  • 若后续无更高气温,值为 0。

问题本质:对数组中每个元素,找到右侧第一个更大元素,并计算下标间距。

二、基础实现:暴力枚举思路与局限

最直观的思路是暴力双重循环:

  • 外层遍历每一天 i
  • 内层从 i+1 向后遍历,找到第一个 j 满足 temperatures[j] > temperatures[i],记录 j-i
  • 遍历结束未找到则赋值 0。
cpp 复制代码
// 暴力实现(示意)
vector<int> dailyTemperatures(vector<int>& temperatures) {
    int n = temperatures.size();
    vector<int> ans(n, 0);
    for (int i = 0; i < n; ++i) {
        for (int j = i + 1; j < n; ++j) {
            if (temperatures[j] > temperatures[i]) {
                ans[i] = j - i;
                break;
            }
        }
    }
    return ans;
}

局限分析

时间复杂度 O(N²):当数组长度较大(如 10⁴ 及以上)时,循环次数会急剧增加,出现明显性能瓶颈;同时内层循环存在大量重复比较,前序元素的比较结果未被复用,计算效率较低。

三、优化方向:状态复用与单调栈引入

暴力解法的核心问题是:每个元素会被多次比较,没有记录「尚未找到更大值的元素」的状态。

观察遍历过程:

  • 我们从左到右遍历,先遍历到的元素,需要等待后遍历的元素来匹配「更大值」
  • 这种「先等待、后匹配」的场景,适合用栈保存未完成匹配的元素状态
  • 为了保证匹配效率,栈需要维持单调递减的特性:栈中元素从栈底到栈顶不递增,确保每次新元素入栈时,能直接找到所有可匹配的栈内元素。

四、单调栈解法:逻辑、实现与复杂度

核心思路

  1. 栈中存储下标而非温度值:既可以通过下标获取温度,又能直接计算等待天数,避免额外存储;
  2. 维护单调递减栈:栈内下标对应的温度,自底向上递减;
  3. 遍历逻辑:
    • 若当前温度 > 栈顶下标对应温度,说明栈顶元素找到「右侧第一个更大值」;
    • 弹出栈顶下标,计算并赋值答案;
    • 重复上述过程,直到栈为空或栈顶温度不小于当前温度,再将当前下标入栈。

完整实现(C++)

cpp 复制代码
class Solution {
public:
    vector<int> dailyTemperatures(vector<int>& temperatures) {
        int n = temperatures.size();
        vector<int> ans(n, 0);
        stack<int> st;

        for (int i = 0; i < n; ++i) {
            // 栈非空且当前温度更高,匹配栈顶元素
            while (!st.empty() && temperatures[st.top()] < temperatures[i]) {
                int idx = st.top();
                ans[idx] = i - idx;
                st.pop();
            }
            st.push(i);
        }
        return ans;
    }
};

复杂度分析

  • 时间复杂度 O(N):每个元素仅入栈、出栈各 1 次,while 循环的总执行次数与数组长度线性相关,无冗余计算;
  • 空间复杂度 O(N):最坏情况下(数组严格递减),栈存储全部元素下标,空间与输入规模成正比。

五、工程实现中的细节与考量

  1. 存储下标而非数值

    若存储温度值,还需额外记录对应下标,增加内存开销与寻址步骤;直接存下标,一次存储即可满足「比较温度」「计算天数」两个需求,是更简洁的工程选择。

  2. 栈结构的选型

    标准库 stack 底层默认依赖 deque,若追求更优的内存连续性 ,可改用 vector 模拟栈(用 push_back/pop_back 实现),在高频调用场景下,内存局部性更优。

  3. 边界与特殊情况

    • 空数组、单元素数组:直接返回全 0 数组,代码天然兼容;
    • 重复温度:栈的递减特性可正确处理,仅「严格更大」时匹配,符合题目要求。
  4. 与实际业务场景的适配

    该思路不局限于气温问题,可直接迁移到时序数据(股价、湿度、流量等)的「下一个更大值」场景,是处理单向时序匹配问题的通用线性解法。

六、知识迁移与小结

本题是**单调栈求解「下一个更大/更小元素」**的典型入门题:

  • 核心是用栈保存「未完成匹配的状态」,通过单调特性避免重复比较,将平方复杂度优化为线性;
  • 工程上优先考虑状态复用存储效率,减少冗余计算与内存开销;
  • 掌握这一思路后,可快速解决同类问题(如柱状图中最大矩形、接雨水、下一个更大元素 I/II 等)。
相关推荐
中屹指纹浏览器1 天前
2026基于内核隔离的浏览器环境虚拟化技术在企业数字化运营中的应用研究
经验分享·笔记
一轮弯弯的明月1 天前
Python基础-速通秘籍(下)
开发语言·笔记·python·学习
charlie1145141911 天前
2026年正点原子开发板移植方案——从0开始的Rootfs之路(3)inittab 与 init 系统:Linux 启动的“第一号进程“全解析
linux·驱动开发·学习·嵌入式开发·嵌入式linux
wsx_iot1 天前
TDengine学习
数据库·学习·tdengine
AI成长日志1 天前
【笔面试算法学习专栏】二分查找专题:力扣hot100经典题目深度解析
学习·算法·面试
nqqcat~1 天前
shell入门
笔记
Aaswk1 天前
刷题笔记(回溯算法)
数据结构·c++·笔记·算法·leetcode·深度优先·剪枝
Frostnova丶1 天前
(11)LeetCode 239. 滑动窗口最大值
数据结构·算法·leetcode
m0_564914921 天前
AI学习课堂网站丨OPENMAIC丨清华团队开源项目
学习
参.商.1 天前
【Day48】46. 全排列
leetcode·golang