动态规划填表技巧:固定最后一个数 vs 固定倒数第二个数

在动态规划中,填表时固定最后一个数还是倒数第二个数,取决于问题的定义和状态转移方程的设计。


目录

[1. 固定最后一个数](#1. 固定最后一个数)

适用场景

特点

示例

[2. 固定倒数第二个数](#2. 固定倒数第二个数)

适用场景

特点

示例

[3. 固定最后一个数与倒数第二个数的对比](#3. 固定最后一个数与倒数第二个数的对比)

[4. 总结](#4. 总结)


1. 固定最后一个数

适用场景

  • 当问题的解与以某个位置为结尾的子问题相关时,通常固定最后一个数。

  • 例如:最大子数组和、最长递增子序列(LIS)等问题。

特点

  • 状态定义通常为 dp[i],表示以第 i 个元素为结尾的最优解。

  • 状态转移方程通常依赖于前一个状态(dp[i-1])或前面所有状态。

示例

  • 最大子数组和

    状态定义:dp[i] 表示以 nums[i] 为结尾的最大子数组和。

    状态转移:dp[i] = max(nums[i], dp[i-1] + nums[i])

    代码:

    cpp 复制代码
    int maxSubArray(vector<int>& nums) {
        int dp = nums[0], res = dp;
        for (int i = 1; i < nums.size(); i++) {
            dp = max(nums[i], dp + nums[i]);
            res = max(res, dp);
        }
        return res;
    }
  • 最长递增子序列(LIS)

    状态定义:dp[i] 表示以 nums[i] 为结尾的最长递增子序列长度。

    状态转移:dp[i] = max(dp[i], dp[j] + 1),其中 j < inums[j] < nums[i]

    代码:

    cpp 复制代码
    int lengthOfLIS(vector<int>& nums) {
        vector<int> dp(nums.size(), 1);
        int res = 1;
        for (int i = 1; i < nums.size(); i++) {
            for (int j = 0; j < i; j++) {
                if (nums[i] > nums[j]) {
                    dp[i] = max(dp[i], dp[j] + 1);
                }
            }
            res = max(res, dp[i]);
        }
        return res;
    }

2. 固定倒数第二个数

适用场景

  • 当问题的解与子问题的中间状态相关时,通常固定倒数第二个数。

  • 例如:最长公共子序列(LCS)、编辑距离等问题。

特点

  • 状态定义通常为 dp[i][j],表示前 i 个元素和前 j 个元素之间的某种关系。

  • 状态转移方程通常依赖于 dp[i-1][j-1]dp[i-1][j]dp[i][j-1]

示例

  • 最长公共子序列(LCS)

    状态定义:dp[i][j] 表示字符串 A 的前 i 个字符和字符串 B 的前 j 个字符的最长公共子序列长度。

    状态转移:

    cpp 复制代码
    if (A[i-1] == B[j-1]) {
        dp[i][j] = dp[i-1][j-1] + 1;
    } else {
        dp[i][j] = max(dp[i-1][j], dp[i][j-1]);
    }

    代码:

    cpp 复制代码
    int longestCommonSubsequence(string text1, string text2) {
        int m = text1.size(), n = text2.size();
        vector<vector<int>> dp(m + 1, vector<int>(n + 1, 0));
        for (int i = 1; i <= m; i++) {
            for (int j = 1; j <= n; j++) {
                if (text1[i-1] == text2[j-1]) {
                    dp[i][j] = dp[i-1][j-1] + 1;
                } else {
                    dp[i][j] = max(dp[i-1][j], dp[i][j-1]);
                }
            }
        }
        return dp[m][n];
    }
  • 编辑距离

    状态定义:dp[i][j] 表示将字符串 A 的前 i 个字符转换为字符串 B 的前 j 个字符所需的最小操作次数。

    状态转移:

    cpp 复制代码
    if (A[i-1] == B[j-1]) {
        dp[i][j] = dp[i-1][j-1];
    } else {
        dp[i][j] = min({dp[i-1][j], dp[i][j-1], dp[i-1][j-1]}) + 1;
    }

    代码:

    cpp 复制代码
    int minDistance(string word1, string word2) {
        int m = word1.size(), n = word2.size();
        vector<vector<int>> dp(m + 1, vector<int>(n + 1, 0));
        for (int i = 0; i <= m; i++) dp[i][0] = i;
        for (int j = 0; j <= n; j++) dp[0][j] = j;
        for (int i = 1; i <= m; i++) {
            for (int j = 1; j <= n; j++) {
                if (word1[i-1] == word2[j-1]) {
                    dp[i][j] = dp[i-1][j-1];
                } else {
                    dp[i][j] = min({dp[i-1][j], dp[i][j-1], dp[i-1][j-1]}) + 1;
                }
            }
        }
        return dp[m][n];
    }

3. 固定最后一个数与倒数第二个数的对比

特性 固定最后一个数 固定倒数第二个数
适用问题 子数组、子序列问题 双序列、矩阵类问题
状态定义 dp[i]dp[i][j] dp[i][j]
状态转移 依赖前一个状态或前面所有状态 依赖 dp[i-1][j-1] 等中间状态
典型问题 最大子数组和、LIS LCS、编辑距离

4. 总结

  • 固定最后一个数:适用于子数组或子序列问题,状态转移通常依赖于前一个状态或前面所有状态。

  • 固定倒数第二个数 :适用于双序列或矩阵类问题,状态转移通常依赖于中间状态(如 dp[i-1][j-1])。

相关推荐
菜鸟破茧计划9 分钟前
线段树:数据结构中的超级英雄
数据结构·c++·算法
秦少游在淮海9 分钟前
leetcode - 双指针问题
算法·leetcode
光电大美美-见合八方中国芯14 分钟前
【平面波导外腔激光器专题系列】1064nm单纵模平面波导外腔激光器‌
网络·数据库·人工智能·算法·平面·性能优化
iceslime28 分钟前
算法设计与分析实验题-序列对齐
数据结构·c++·算法·算法设计与分析·序列对齐
羊小猪~~35 分钟前
深度学习基础--目标检测常见算法简介(R-CNN、Fast R-CNN、Faster R-CNN、Mask R-CNN、SSD、YOLO)
人工智能·深度学习·算法·yolo·目标检测·机器学习·cnn
多多*1 小时前
分布式ID设计 数据库主键自增
数据库·sql·算法·http·leetcode·oracle
D_aniel_1 小时前
排序算法-希尔排序
java·算法·排序算法·希尔排序
CodeWithMe2 小时前
【C/C++】C++中noexcept的妙用与性能提升
c语言·开发语言·c++
非著名架构师2 小时前
C++跨平台开发实践:深入解析与常见问题处理指南
开发语言·c++
SuperCandyXu2 小时前
leetcode0310. 最小高度树-medium
数据结构·c++·算法·leetcode