线性动态规划问题
创始人
2025-05-31 12:08:16
0

文章目录

    • 1. 三角形中最小路径之和
    • 2. 最长递增子序列
    • 3. 最长公共子序列

1. 三角形中最小路径之和

给定一个三角形 triangle ,找出自顶向下的最小路径和。每一步只能移动到下一行中相邻的结点上。相邻的结点 在这里指的是 下标 与 上一层结点下标 相同或者等于 上一层结点下标 + 1 的两个结点。也就是说,如果正位于当前行的下标 i ,那么下一步可以移动到下一行的下标 i 或 i + 1 。

示例:
在这里插入图片描述

分析:

在这里插入图片描述
一般涉及到i-1的下标,我们i的取值从1开始。
动态规划问题的时间复杂度一般为:状态数量*转移计算量

  • 二维数组:自顶向下
class Solution {
public:int minimumTotal(vector>& triangle) {int dp[201][201];int n = triangle.size();//不能使用int dp[201][201] = {INT_MAX},因为这个仅仅是把dp[0][0] = INT_MAX,其余还是0for (int i = 0; i < 201; ++i) {for (int j = 0; j < 201; ++j) {dp[i][j] = INT_MAX;}}dp[0][0] = triangle[0][0];int minpath = INT_MAX;for(int i = 1; i < n; ++i){for(int j = 0; j <= i; ++j){if(j > 0)   dp[i][j] = min(dp[i - 1][j - 1], dp[i - 1][j]) + triangle[i][j];else    dp[i][j] = dp[i - 1][j] + triangle[i][j];}}for(int i = 0; i < n; ++i){minpath = min(minpath, dp[n - 1][i]);}return minpath;}
};

时间复杂度:O(n^2), 空间复杂度:O(n^2)

  • 一维数组:自底向上
class Solution {
public:int minimumTotal(vector>& triangle){int dp[201];int n = triangle.size() - 1;for(int i = 0; i <= n; ++i){dp[i] = triangle[n][i];}for(int i = n - 1; i >= 0; --i){for(int j = 0; j <= i; ++j){dp[j] = min(dp[j] + triangle[i][j], dp[j + 1] + triangle[i][j]);}}return dp[0];}
};

时间复杂度:O(n^2),空间复杂度:O(n)

2. 最长递增子序列

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

分析
在这里插入图片描述

class Solution {
public:int lengthOfLIS(vector& nums) {int dp[2501];for(int i = 0; i < nums.size(); ++i){dp[i] = 1;for(int j = 0; j < i; ++j){if(nums[i] > nums[j])   dp[i] = max(dp[i], dp[j] + 1);}}int result = 0;for(int i = 0; i < nums.size(); ++i) result = max(result, dp[i]);return result;}
};

如何保存最长递增子序列

int lengthOfLIS(vector& nums) {int dp[2501];int g[2501]; //记录最长子序列for (int i = 0; i < nums.size(); ++i) {dp[i] = 1;g[i] = 0;for (int j = 0; j < i; ++j) {if (nums[i] > nums[j]) {if (dp[i] < dp[j] + 1) {dp[i] = dp[j] + 1;//记录dp[i]从哪个状态转移过来的g[i] = j;}}}}int result = 0;int k = 0;for (int i = 0; i < nums.size(); ++i) {if (dp[k] < dp[i]) {k = i;}}result = dp[k];//倒着输出,如果需要正着输出只需要逆序就可以for (int i = 0; i < result; ++i) {cout << nums[k] << " ";k = g[k];}cout << endl;return result;
}
int main()
{vector num = {0,1,0,3,2,3};cout<

在这里插入图片描述

3. 最长公共子序列

给定两个字符串 text1 和 text2,返回这两个字符串的最长 公共子序列 的长度。如果不存在 公共子序列 ,返回 0 。一个字符串的 子序列 是指这样一个新的字符串:它是由原字符串在不改变字符的相对顺序的情况下删除某些字符(也可以不删除任何字符)后组成的新字符串。

例如,“ace” 是 “abcde” 的子序列,但 “aec” 不是 “abcde” 的子序列。两个字符串的 公共子序列 是这两个字符串所共同拥有的子序列。

分析:主要就是两大情况: text1[i - 1] 与 text2[j - 1]相同,text1[i - 1] 与 text2[j - 1]不相同

  • 如果text1[i - 1] 与 text2[j - 1]相同,那么找到了一个公共元素,所以dp[i][j] = dp[i - 1][j - 1] + 1;
  • 如果text1[i - 1] 与 text2[j - 1]不相同,那就看看text1[0, i - 2]与text2[0, j - 1]的最长公共子序列 和 text1[0, i - 1]与text2[0, j - 2]的最长公共子序列,取最大的。即:dp[i][j] = max(dp[i - 1][j], dp[i][j - 1]);
    在这里插入图片描述
class Solution {
public:/*//递归实现会超时int longestCommonSubsequence(string text1,int n,string text2,int m){if(n < 0 || m < 0){return 0;}if(dp[n][m] >= 0){return dp[n][m];}if(text1[n] == text2[m]){dp[n][m] = 1 + longestCommonSubsequence(text1,n-1,text2,m-1);}else{int l1 = longestCommonSubsequence(text1,n-1,text2,m);int l2 = longestCommonSubsequence(text1,n,text2,m-1);dp[n][m] = max(l1,l2);}return dp[n][m];}int longestCommonSubsequence(string text1, string text2) {dp.resize(text1.size(),vector(text2.size(),-1));return longestCommonSubsequence(text1,text1.size()-1,text2,text2.size()-1);}
private:vector> dp;*///动态规划int longestCommonSubsequence(string text1, string text2) {int i = text1.size();int j = text2.size();vector> dp(i+1,vector(j+1,0));for(int n = 1;n <= i;n++){for(int m = 1;m <= j;m++){if(text1[n-1] == text2[m-1]){dp[n][m] = 1 + dp[n-1][m-1];}else{dp[n][m] = max(dp[n-1][m],dp[n][m-1]);}}}return dp[i][j];}
};

相关内容

热门资讯

Android开发-Andro... 01  Android UI 1.1  UI 用户界面(User Interface,...
有关于致父母的一封感谢信 给父... 父母的教育素质影响着学前儿童家庭教育的效果和质量,也直接决定着家庭教育的优劣成败。那么有关于致父母的...
有关于写给老师的一封感谢信 给... 感谢信作为一种礼貌性言语行为的载体,其中谢意表达的处理成为写好感谢信的核心环节。下面小编整理了有关于...
有关于致干部家属的感谢信 致干... 感谢信是写信人对给予自己帮助、支持、关心、祝贺、慰问、馈赠的施惠人表示答谢而写的社交礼仪书信。下面小...
婚礼主持词开场白,浪漫婚礼主持... 1、开场白:(神秘花园)  我们每个人都生长在同一个蔚蓝的星球;我们每个人都是从平坦或不平坦的一端开...
有关于致公交司机的感谢信 给公... 公交司机作为影响城市公共交通供给质量和服务满意度的双重介入主体,其个体行为在很大程度上代表了城市综合...
A.机器学习入门算法(三):基... 机器学习算法(三):K近邻(k-nearest neigh...
有关于写给公安机关的感谢信 写... 公安机关内部管理制度是否完善,对于确保公安队伍建设和管理,保证公安机关顺利地履行职责和任务,具有重大...
JAVA多线程知识整理 Java多线程基础 线程的创建和启动 继承Thread类来创建并启动 自定义Thread类的子类&#...
有关于写给公安局的一封感谢信 ... 公安局对固定资产指标的要求也越来越多,固定资产管理工作成为公安局日常一项很重要的工作。那么xxx应该...
有关于致公交公司的感谢信 给公... 公交公司的生产经营活动不仅具有经济效益,而且具有社会效益。那么xxx应该怎么写呢?下面小编整理了有关...
有关于优秀员工致公司的感谢信 ... 有对自然物象的感谢,对伟大人物的感谢,更有对日常生活的感谢。那么关于优秀员工致公司的感谢信应该怎么写...
有关于员工致公司领导的感谢信 ... 谢语是以表达感谢为要点。那么有关于员工致公司领导的感谢信应该怎么写呢?下面小编整理了有关于员工致公司...
药品批准文号查询|药融云-中国... 药品批文是国家食品药品监督管理局(NMPA)对药品的审评和批准的证明文件...
最新或2023(历届)给公司领...  给公司领导的感谢信范文一:  敬爱的领导:  你们好,回想在港城办事处三年多的日子里,我得到了各位...
最新或2023(历届)企业致客...   企业致客户的感谢信范文一:  尊敬的客户:  您好!  新的一年就要到来了,怀着感恩的心情,向您...
给公司捐款的感谢信范文 公司捐...  给公司捐款的感谢信范文一:  尊敬的领导、工友们:  你们好!首先我想表达我们一家人最诚挚的谢意和...
python 基础系列篇:四、... python 基础系列篇:四、编写两个简单的小游戏(猜数字及2048&#...
写给居委会的感谢信范文 怎么给... 写给居委会的感谢信范文一:  尊敬的上汽社区居委会领导们:  您们好!  首先感谢上汽社区居委会的各...
最新或2023(历届)终给客户...  年终给客户的感谢信范文一:  尊敬的客户:  您好!新的一年即将到来,我们怀着感恩的心情向您致以亲...