线性动态规划问题
创始人
2025-05-31 11:01:49
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];}
};

相关内容

热门资讯

Java培训课程大纲对于我们来... 不少同学想要参加Java培训在最短的时间内让自己从零基础达到一个合格的Java程序员,...
烤了10年披萨,这面团最松软,... #图文打卡计划#每次路过披萨店,那股浓郁的芝士香总能让人走不动道。特别是看到热乎乎的披萨拉出长长的丝...
吕文扬的元诚樱桃干试吃记 初夏的午后,阳光透过玻璃窗洒在桌面上,映出一片温暖的金色。吕文扬坐在桌前,面前摆着一包刚刚拆封的樱桃...
炖鸡汤的秘诀:营养满满,口感醇... 在寒冷的冬季,一碗热腾腾的鸡汤不仅能温暖身心,还能滋补身体。今天,我将与大家分享如何炖出好喝又营养的...
蒸馒头,直接上锅是大错!塌陷变... 每次掀开蒸锅,看到塌陷变硬的馒头,是不是特别挫败?明明揉面时那么卖力,出锅却像泄了气的皮球。别急着怪...
怪味胡豆在家做!香脆过瘾的川味... 嘴馋的时候,总想找点香香脆脆的小零嘴解解闷?别再盯着包装袋里的薯片了!今天教你做一道地道的重庆风味小...
玫瑰锅炸制作要领 粉丝朋友: 您好,馔墨斋主!我最近想做一道川菜传统甜品——玫瑰锅炸,但听说制作锅炸坯子的技术性很强,...
活动集萃 | 连图举办六一亲子... 端午安康 粽叶飘香 — DRAGON BOAT FESTIVAL — 端午安康·粽叶飘香 为传承中...
鲜嫩山药炒肉片,健康又美味的秘... 哇塞!这道山药炒肉片,香到邻居都来敲门啦! 宝子们,今天给大家分享一道超绝的家常菜——山药炒肉片!这...
iPerf3 -w 参数详细图... 本文目录1、 官方解释2、-w参数使用3、具体参数使用说明3.1、对于UDP,-w的使...
去四川旅游攻略报团五天四晚多少... 标题:去四川旅游攻略报团五天四晚多少钱?驴友亲测,乐乐带你玩转天府之国! 四川旅游推荐!当地导游-乐...
四川九寨沟都江堰旅游纯玩团5天... 四川九寨沟都江堰旅游纯玩团5天4晚攻略及报价,驴友亲测! 四川旅游推荐!当地导游-乐乐:185 83...
琴岛观澜丨山海之城的长红之道 齐鲁晚报·齐鲁壹点 杨雪 当华晨宇演唱会的音浪撞上小麦岛的海风,当《送你一朵小红花》的镜头扫过 “孤...
四川旅游攻略自由行攻略旅行团五... 标题:四川五日游亲测攻略,跟着本地团导游乐乐畅享天府之国! 四川旅游推荐!当地导游-乐乐:185 8...
Java多线程之Executo... 文章目录1 ExecutorCompletionService1.1 简介1.2 原理1.3 Dem...
首个上美影IP主题乐园登陆上海... 国内首个以上美影经典动画IP为核心的沉浸式儿童益智乐园,在六一国际儿童节到来之际正式开园。乐园占地面...
Linux学习之端口、网络协议... 端口:设备与外界通讯交流的出口 网络协议:   网络协议是指计算机通信网...
基于 Zynq+AD+DA 的... 4 振动台控制算法的 FPGA 实现 4.1 PID 控制算法 4.1.1 增量式 PID 控制...
陕西发布26条夏季乡村休闲旅游... 5月29日,陕西省农业农村厅以“夏纳凉 享田园惬意时光”为主题,向社会推介发布26条夏季乡村休闲旅游...
超详细-安装vCenterv ... 目录 介绍: 第一阶段安装: 第二阶段安装: 最近在玩虚拟...