学习笔记|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 等)。
相关推荐
Navigator_Z5 小时前
LeetCode //C - 1089. Duplicate Zeros
c语言·算法·leetcode
sulikey7 小时前
个人Linux操作系统学习笔记6 - 操作系统与进程初识
linux·笔记·学习·操作系统·进程
unicorn318 小时前
学习学习学习
学习
XGeFei8 小时前
【Fastapi学习笔记(3)】——资源的层级关系、安全性-幂等性、Field、工厂函数
笔记·学习·fastapi
语戚9 小时前
力扣 3161. 块放置查询:线段树解法(Java 实现)
java·算法·leetcode·面试·线段树·力扣·
星恒随风9 小时前
Python 基础语法详解(一):从表达式、变量到数据类型
开发语言·笔记·python·学习
暴躁小师兄数据学院11 小时前
【AI大数据工程师特训笔记】第14讲:Linux操作系统与shell脚本
大数据·人工智能·笔记
8Qi811 小时前
LeetCode 23. 合并 K 个升序链表 —— 小顶堆(PriorityQueue)
数据结构·算法·leetcode·链表·
tedcloud12311 小时前
cc-switch评测:多AI Coding Agent管理工具详解
数据库·人工智能·sql·学习·自动化
土狗TuGou11 小时前
SQL内功笔记 · 第8篇:事务的四大特性与隔离级别
数据库·笔记·后端·sql·mysql·oracle