动态规划经典题解:最长递增子序列 & 乘积最大子数组

目录

一、最长递增子序列(LIS)

题目描述

思路分析

[1. 动态规划解法(基础版)](#1. 动态规划解法(基础版))

[2. 优化解法(贪心 + 二分)](#2. 优化解法(贪心 + 二分))

代码实现(Java)

二、乘积最大子数组

题目描述

思路分析

代码实现(Java)

三、两道题的核心对比与面试考点

四、总结


大家好!今天我们来拆解两道动态规划的经典中等难度题目:最长递增子序列乘积最大子数组。这两道题都是大厂面试里的高频考点,而且非常能体现动态规划的核心思想 ------ 如何通过子问题的最优解,一步步推导出全局最优解。


一、最长递增子序列(LIS)

题目描述

给你一个整数数组 nums,找到其中最长严格递增子序列的长度。子序列是由数组派生而来的序列,删除(或不删除)数组中的元素而不改变其余元素的顺序。例如,[3,6,2,7] 是数组 [0,3,1,6,2,2,7] 的子序列。

思路分析

1. 动态规划解法(基础版)
  • 定义状态dp[i] 表示以 nums[i] 结尾的最长递增子序列的长度。
  • 状态转移方程 :对于每个 i,遍历 0i-1 的所有 j:如果 nums[i] > nums[j],则 dp[i] = max(dp[i], dp[j] + 1)
  • 初始状态 :每个元素自身都是一个长度为 1 的子序列,所以 dp[i] = 1
  • 结果dp 数组中的最大值,就是整个数组的最长递增子序列长度。
2. 优化解法(贪心 + 二分)

动态规划的时间复杂度是 O(n2),我们可以用贪心 + 二分将复杂度优化到 O(nlogn)。

  • 维护一个数组 tails,其中 tails[k] 表示长度为 k+1 的最长递增子序列的最小可能的末尾元素。
  • 遍历 nums 中的每个数 x
    • 如果 x 大于 tails 最后一个元素,直接追加到 tails 末尾;
    • 否则,找到 tails 中第一个大于等于 x 的位置,替换为 x
  • 最终 tails 的长度就是 LIS 的长度。

代码实现(Java)

java

运行

复制代码
// 动态规划 O(n²) 版本
public int lengthOfLIS(int[] nums) {
    if (nums == null || nums.length == 0) return 0;
    int[] dp = new int[nums.length];
    Arrays.fill(dp, 1);
    int maxLen = 1;
    for (int i = 1; i < nums.length; i++) {
        for (int j = 0; j < i; j++) {
            if (nums[i] > nums[j]) {
                dp[i] = Math.max(dp[i], dp[j] + 1);
            }
        }
        maxLen = Math.max(maxLen, dp[i]);
    }
    return maxLen;
}

// 贪心 + 二分 O(n log n) 版本
public int lengthOfLISOpt(int[] nums) {
    if (nums == null || nums.length == 0) return 0;
    List<Integer> tails = new ArrayList<>();
    for (int x : nums) {
        if (tails.isEmpty() || x > tails.get(tails.size() - 1)) {
            tails.add(x);
        } else {
            // 二分查找第一个 >= x 的位置
            int left = 0, right = tails.size() - 1;
            while (left < right) {
                int mid = (left + right) / 2;
                if (tails.get(mid) >= x) {
                    right = mid;
                } else {
                    left = mid + 1;
                }
            }
            tails.set(left, x);
        }
    }
    return tails.size();
}

二、乘积最大子数组

题目描述

给你一个整数数组 nums,请你找出数组中乘积最大的非空连续子数组(该子数组中至少包含一个数字),并返回该子数组所对应的乘积。

思路分析

这道题和经典的「最大子数组和」很像,但乘积有个特殊的问题:负数的存在会让最小值变成最大值 。比如:当前是 -2,后面跟着一个 -3,那么 -2 * -3 = 6 就会变成正数,比之前的正数乘积更大。

  • 定义状态

    • maxDp[i]:以 nums[i] 结尾的最大乘积子数组的乘积。
    • minDp[i]:以 nums[i] 结尾的最小乘积子数组的乘积。
  • 状态转移方程

    plaintext

    复制代码
    maxDp[i] = max(nums[i], maxDp[i-1] * nums[i], minDp[i-1] * nums[i])
    minDp[i] = min(nums[i], maxDp[i-1] * nums[i], minDp[i-1] * nums[i])

    因为 nums[i] 可能是负数,乘以之前的最小值反而可能得到最大值,所以必须同时维护最大和最小值。

  • 初始状态maxDp[0] = minDp[0] = nums[0]

  • 结果 :遍历 maxDp 数组,取最大值即可。

代码实现(Java)

java

运行

复制代码
public int maxProduct(int[] nums) {
    if (nums == null || nums.length == 0) return 0;
    int n = nums.length;
    int maxDp = nums[0];
    int minDp = nums[0];
    int result = nums[0];
    for (int i = 1; i < n; i++) {
        // 必须先保存,避免被覆盖
        int currMax = maxDp;
        int currMin = minDp;
        maxDp = Math.max(nums[i], Math.max(currMax * nums[i], currMin * nums[i]));
        minDp = Math.min(nums[i], Math.min(currMax * nums[i], currMin * nums[i]));
        result = Math.max(result, maxDp);
    }
    return result;
}

三、两道题的核心对比与面试考点

表格

题目 核心难点 优化技巧 面试常见追问
最长递增子序列 如何定义状态、如何优化时间复杂度 贪心 + 二分,将 O(n2) 优化到 O(nlogn) 请输出最长递增子序列本身,而不是长度
乘积最大子数组 负数导致的 "最大 / 最小值反转" 问题 同时维护 maxmin 两个状态变量 如果数组全是负数,怎么处理?如果包含 0 呢?

四、总结

  1. 最长递增子序列
    • 基础版 DP 是理解动态规划状态定义和转移的绝佳例子。
    • 优化版的贪心 + 二分,体现了如何用额外的空间和更高效的算法,突破 DP 的时间复杂度瓶颈。
  2. 乘积最大子数组
    • 核心是打破思维定势,不像 "最大子数组和" 只维护一个最大值,而是要同时维护最大和最小值,因为负数的存在会让最小值变成最大值。
相关推荐
洛水水1 小时前
【力扣100题】18.随机链表的复制
算法·leetcode·链表
南宫萧幕1 小时前
规则基 EMS 仿真实战:SOC 区间划分与 Simulink 闭环建模全解
算法·matlab·控制
多加点辣也没关系2 小时前
数据结构与算法|第二十三章:高级数据结构
数据结构·算法
hoiii1874 小时前
孤立森林 (Isolation Forest) 快速异常检测系统
算法
c++之路5 小时前
适配器模式(Adapter Pattern)
java·算法·适配器模式
吴声子夜歌5 小时前
Java——接口的细节
java·开发语言·算法
myheartgo-on5 小时前
Java—方 法
java·开发语言·算法·青少年编程
庞轩px6 小时前
第六篇:Spring用了哪些设计模式?——从单例到代理,拆解框架中的经典设计
java·spring·设计模式·bean·代理模式·aop·单例
宝贝儿好7 小时前
【LLM】第三章:项目实操案例:智能输入法项目
人工智能·python·深度学习·算法·机器人