二刷 LeetCode:118. 杨辉三角 & 198. 打家劫舍 复盘笔记

目录

[一、118. 杨辉三角](#一、118. 杨辉三角)

题目回顾

思路复盘

代码实现(Java)

[易错点 & 二刷心得](#易错点 & 二刷心得)

[二、198. 打家劫舍](#二、198. 打家劫舍)

题目回顾

思路复盘

[基础 DP 实现(Java)](#基础 DP 实现(Java))

[空间优化版(O (1) 空间)](#空间优化版(O (1) 空间))

[易错点 & 二刷心得](#易错点 & 二刷心得)

[三、两道题的共性总结 & 二刷收获](#三、两道题的共性总结 & 二刷收获)


这两道题分别是动态规划入门经典一维 DP的代表,也是面试高频考点。二刷时我们重点拆解思路、优化写法,顺便把易错点和通用模板总结清楚。


一、118. 杨辉三角

题目回顾

给定一个非负整数 numRows,生成「杨辉三角」的前 numRows 行。在「杨辉三角」中,每个数是它左上方和右上方的数的和。

思路复盘

杨辉三角的核心规律:

  1. i 行(从 0 开始)有 i+1 个元素
  2. 每行的第一个和最后一个元素都是 1
  3. 中间元素:triangle[i][j] = triangle[i-1][j-1] + triangle[i-1][j]
代码实现(Java)

java

运行

复制代码
public List<List<Integer>> generate(int numRows) {
    List<List<Integer>> result = new ArrayList<>();
    for (int i = 0; i < numRows; i++) {
        List<Integer> row = new ArrayList<>();
        // 每行首尾为1
        for (int j = 0; j <= i; j++) {
            if (j == 0 || j == i) {
                row.add(1);
            } else {
                // 中间元素 = 上一行的两个元素之和
                row.add(result.get(i-1).get(j-1) + result.get(i-1).get(j));
            }
        }
        result.add(row);
    }
    return result;
}

易错点 & 二刷心得

  1. 索引处理 :注意行号从 0 开始,第 i 行的长度是 i+1,避免越界。
  2. 边界元素 :每行首尾元素必须单独处理为 1,否则会出现 i-1j-1 为负的越界错误。
  3. 空间优化 :如果题目只要求返回第 k 行,可以用一维数组滚动更新,空间复杂度从 O (n²) 降到 O (n)。

二、198. 打家劫舍

题目回顾

你是一个专业的小偷,计划偷窃沿街的房屋。每间房内都藏有一定的现金,影响你偷窃的唯一制约因素就是相邻的房屋装有相互连通的防盗系统,如果两间相邻的房屋在同一晚上被小偷闯入,系统会自动报警。给定一个代表每个房屋存放金额的非负整数数组,计算你 不触动警报装置的情况下 ,一夜之内能够偷窃到的最高金额。

思路复盘

这是一维动态规划的经典题,核心是定义状态和状态转移方程。

  1. 状态定义dp[i] 表示前 i 间房屋能偷窃到的最高金额。
  2. 状态转移
    • 对于第 i 间房屋,有两种选择:
      1. 偷第 i 间:那么不能偷第 i-1 间,最高金额为 dp[i-2] + nums[i]
      2. 不偷第 i 间:最高金额为 dp[i-1]
    • 状态转移方程:dp[i] = max(dp[i-1], dp[i-2] + nums[i])
  3. 初始状态
    • dp[0] = nums[0](只有一间房,偷它)
    • dp[1] = max(nums[0], nums[1])(两间房,偷金额大的)
  4. 结果dp[n-1](n 为房屋数量)
基础 DP 实现(Java)

java

运行

复制代码
public int rob(int[] nums) {
    if (nums == null || nums.length == 0) return 0;
    if (nums.length == 1) return nums[0];
    int n = nums.length;
    int[] dp = new int[n];
    dp[0] = nums[0];
    dp[1] = Math.max(nums[0], nums[1]);
    for (int i = 2; i < n; i++) {
        dp[i] = Math.max(dp[i-1], dp[i-2] + nums[i]);
    }
    return dp[n-1];
}
空间优化版(O (1) 空间)

因为 dp[i] 只依赖 dp[i-1]dp[i-2],所以可以用两个变量滚动更新,空间复杂度从 O (n) 降到 O (1):

java

运行

复制代码
public int rob(int[] nums) {
    if (nums == null || nums.length == 0) return 0;
    if (nums.length == 1) return nums[0];
    int prevPrev = nums[0];
    int prev = Math.max(nums[0], nums[1]);
    for (int i = 2; i < nums.length; i++) {
        int current = Math.max(prev, prevPrev + nums[i]);
        prevPrev = prev;
        prev = current;
    }
    return prev;
}

易错点 & 二刷心得

  1. 边界处理:数组长度为 0 或 1 时要单独判断,避免索引越界。
  2. 状态转移的理解dp[i] 表示前 i 间的最高金额,不是偷第 i 间的最高金额,所以不偷第 i 间时,dp[i] = dp[i-1]
  3. 空间优化技巧:一维 DP 中,如果当前状态只依赖前两个状态,就可以用变量代替数组,大幅降低空间复杂度。

三、两道题的共性总结 & 二刷收获

  1. 动态规划的入门模板
    • 杨辉三角:二维 DP 的基础,从简单的递推关系入手,理解 "上一行推导下一行" 的思路。
    • 打家劫舍:一维 DP 的经典,学会定义状态、找到转移方程,理解 "选 / 不选" 两种决策的逻辑。
  2. 优化意识
    • 从二维数组到一维数组,再到变量滚动更新,体会空间优化的思路。
    • 二刷时不仅要写出正确的代码,还要思考如何优化时间和空间复杂度。
  3. 面试重点
    • 杨辉三角:重点是边界处理和递推关系,常作为 DP 入门题考察。
    • 打家劫舍:重点是状态转移方程和空间优化,后续的环形打家劫舍、打家劫舍 III 都是它的变形题。
相关推荐
深邃-6 小时前
【数据结构与算法】-二叉树(1):树的概念与结构,二叉树的概念与结构
数据结构·算法·链表·二叉树··顺序表
风筝在晴天搁浅6 小时前
手撕归并排序
数据结构·算法·排序算法
70asunflower6 小时前
半导体产业的经济逻辑、技术瓶颈与AI芯片格局:一份学习笔记
人工智能·笔记·学习
lynnlovemin6 小时前
C++高精度加减乘除算法详解
开发语言·c++·算法·高精度
原来是猿6 小时前
算法中 cin/cout 超时?聊聊它与 printf/scanf 的性能差异
算法
老赵聊算法、大模型备案7 小时前
“清朗·整治AI应用乱象”专项行动深度解读:从资质合规视角看AI应用新规
大数据·人工智能·算法·安全·aigc
凉、介7 小时前
C 语言类型强转引发的隐蔽内存破坏问题分析
c语言·开发语言·笔记·学习·嵌入式
Gary Studio7 小时前
安卓HAL AIDL经验笔记
笔记
Hello.Reader7 小时前
算法基础(二)——算法为什么是一种核心技术
算法