当前位置: 首页 > news >正文

C++ 动态规划

子序列子串相关

单个指一个数组或字符串,两个指两个数组或字符串。

最长上升子序列-单个

dp[i]:以下标i为结尾的递增的最长子序列长度

位置i的最长升序子序列等于j从0到i-1各个位置的最长升序子序列 + 1 的最大值。

class Solution {
public:int lengthOfLIS(vector<int>& nums) {int sz=nums.size();// dp[i] i结尾的最长子序列长度// dp[i]=max(dp[i],dp[j]+1)vector<int> dp(sz,1);int res=1;for(int i=1;i<sz;i++){for(int j=0;j<i;j++){if(nums[i]>nums[j])dp[i]=max(dp[j]+1,dp[i]);}res=max(res,dp[i]);}return res;}
};

最长上升子序列的个数-单个 

最长连续上升子序列-单个

dp[i]:以下标i为结尾的连续递增的子序列长度。

因为本题要求连续递增子序列,所以就只要比较nums[i]与nums[i - 1],而不用去比较nums[j]与nums[i] (j是在0到i之间遍历)。

class Solution {
public:int findLengthOfLCIS(vector<int>& nums) {int sz=nums.size();vector<int> dp(sz,1);int res=1;for(int i=1;i<sz;i++){if(nums[i]>nums[i-1])dp[i]=dp[i-1]+1;res=max(dp[i],res);}return res;}
};

最长重复连续子序列-两个

dp[i][j] :以下标i - 1为结尾的A和以下标j - 1为结尾的B,最长重复子数组长度。

class Solution {
public:int findLength(vector<int>& nums1, vector<int>& nums2) {int sz1=nums1.size(),sz2=nums2.size();vector<vector<int>> dp(sz1+1,vector<int>(sz2+1));int res=0;for(int i=1;i<=sz1;i++){for(int j=1;j<=sz2;j++){if(nums1[i-1]==nums2[j-1])dp[i][j]=dp[i-1][j-1]+1;res=max(res,dp[i][j]);}}return res;}
};

最长重复子序列-两个

dp[i][j]:长度为[0, i - 1]的字符串text1与长度为[0, j - 1]的字符串text2的最长公共子序列。

主要就是两大情况: 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, string text2) {int sz1=text1.size(),sz2=text2.size();int res=0;vector<vector<int>> dp(sz1+1,vector<int>(sz2+1));for(int i=1;i<=sz1;i++){for(int j=1;j<=sz2;j++){if(text1[i-1]==text2[j-1])dp[i][j]=dp[i-1][j-1]+1;elsedp[i][j]=max(dp[i-1][j],dp[i][j-1]);res=max(res,dp[i][j]);}}return res;}
};

不相交的线🧵-两个

直线不能相交,这就是说明在字符串A中 找到一个与字符串B相同的子序列,且这个子序列不能改变相对顺序,只要相对顺序不改变,链接相同数字的直线就不会相交。那么就和最长重复子序列一样了!

最大连续子序列的和-单个

具有最大和的连续子数组。

dp[i]:包括下标i(以nums[i]为结尾)的最大连续子序列和.

dp[i]只有两个方向可以推出来:

  • dp[i - 1] + nums[i],即:nums[i]加入当前连续子序列和
  • nums[i],即:从头开始计算当前连续子序列和

dp[i] = max(dp[i - 1] + nums[i], nums[i]).

class Solution {
public:int maxSubArray(vector<int>& nums) {int sz=nums.size();vector<int> dp(sz);dp[0]=nums[0];int res=nums[0];for(int i=1;i<sz;i++){dp[i]=max(nums[i],dp[i-1]+nums[i]);res=max(res,dp[i]);}return res;}
};

判断子序列-两个

dp[i][j] 表示以下标i-1为结尾的字符串s,和以下标j-1为结尾的字符串t,相同子序列的长度

  • if (s[i - 1] == t[j - 1])
    • t中找到了一个字符在s中也出现了
  • if (s[i - 1] != t[j - 1])
    • 相当于t要删除元素,继续匹配

if (s[i - 1] == t[j - 1]),那么dp[i][j] = dp[i - 1][j - 1] + 1;,因为找到了一个相同的字符,相同子序列长度自然要在dp[i-1][j-1]的基础上加1(如果不理解,在回看一下dp[i][j]的定义

if (s[i - 1] != t[j - 1]),此时相当于t要删除元素,t如果把当前元素t[j - 1]删除,那么dp[i][j] 的数值就是 看s[i - 1]与 t[j - 2]的比较结果了,即:dp[i][j] = dp[i][j - 1]。

class Solution {
public:bool isSubsequence(string s, string t) {int sz1=s.size(),sz2=t.size();vector<vector<int>> dp(sz1+1,vector<int>(sz2+1));for(int i=1;i<=sz1;i++){for(int j=1;j<=sz2;j++){if(s[i-1]==t[j-1]){dp[i][j]=dp[i-1][j-1]+1;}else{dp[i][j]=dp[i][j-1];}}}return dp[sz1][sz2]==sz1;}
};

子序列出现的个数-两个?

dp[i][j]:以i-1为结尾的s子序列中出现以j-1为结尾的t的个数。

回文子串的个数-单个

计算这个字符串中有多少个回文子串。

判断一个子字符串(字符串下标范围[i,j])是否回文

                                               依赖于

子字符串(下标范围[i + 1, j - 1])) 是否是回文

所以为了明确这种递归关系,dp数组是要定义成一个二维dp数组。

布尔类型的dp[i][j]:表示区间范围[i,j] (注意是左闭右闭)的子串是否是回文子串,如果dp[i][j]为true,否则为false。

当s[i]与s[j]不相等,dp[i][j]一定是false。

当s[i]与s[j]相等时,有如下三种情况

  • 情况一:下标i 与 j相同,同一个字符例如a,当然是回文子串
  • 情况二:下标i 与 j相差为1,例如aa,也是回文子串
  • 情况三:下标:i 与 j相差大于1的时候,例如cabac,此时s[i]与s[j]已经相同了,我们看i到j区间是不是回文子串就看aba是不是回文就可以了,那么aba的区间就是 i+1 与 j-1区间,这个区间是不是回文就看dp[i + 1][j - 1]是否为true。
class Solution {
public:int countSubstrings(string s) {int sz=s.size();int res=0;vector<vector<bool>> dp(sz,vector<bool>(sz));for(int i=0;i<sz;i++)dp[i][i]=true;// 注意遍历顺序for(int i=sz-1;i>=0;i--){for(int j=i;j<sz;j++){if(s[i]==s[j]){if(j-i<=1)  dp[i][j]=true;else dp[i][j]=dp[i+1][j-1];}if(dp[i][j])    res++;}}return res;}
};

最长回文子串-单个

class Solution {
public:string longestPalindrome(string s) {int len = 0;int start = 0;int sz = s.size();vector<vector<bool>> dp(sz, vector<bool>(sz));for (int i = sz - 1; i >= 0; i--) {for (int j = i; j < sz; j++) {if (s[i] == s[j]) {if (j - i < 2)dp[i][j] = true;elsedp[i][j] = dp[i + 1][j - 1];}if (dp[i][j]) {if (j - i + 1 > len) {start = i;len = j - i + 1;}}}}return s.substr(start,len);}
};

最长回文子序列-单个

dp[i][j]:字符串s在[i, j]范围内最长的回文子序列的长度

如果s[i]与s[j]相同,那么dp[i][j] = dp[i + 1][j - 1] + 2;

如果s[i]与s[j]不相同,说明s[i]和s[j]的同时加入 并不能增加[i,j]区间回文子序列的长度,那么分别加入s[i]、s[j]看看哪一个可以组成最长的回文子序列。那么dp[i][j]一定是取最大的,即:dp[i][j] = max(dp[i + 1][j], dp[i][j - 1])。

class Solution {
public:int longestPalindromeSubseq(string s) {int sz=s.size();vector<vector<int>> dp(sz,vector<int>(sz));int res=1;for(int i=0;i<sz;i++)dp[i][i]=1;for(int i=sz-1;i>=0;i--){for(int j=i+1;j<sz;j++){if(s[i]==s[j])  dp[i][j]=dp[i+1][j-1]+2;else    dp[i][j]=max(dp[i+1][j],dp[i][j-1]);res=max(res,dp[i][j]);}}return res;}
};

二维动态

不同路径

不同路径II

01背包🎒

有N件物品和一个最多能背重量为W 的背包。第i件物品的重量是weight[i],得到的价值是value[i] 。每件物品只能用一次,求解将哪些物品装入背包里物品价值总和最大。

  • 不放物品i:由dp[i - 1][j]推出,即背包容量为j,里面不放物品i的最大价值,此时dp[i][j]就是dp[i - 1][j]。(其实就是当物品i的重量大于背包j的重量时,物品i无法放进背包中,所以背包内的价值依然和前面相同。)
  • 放物品i:由dp[i - 1][j - weight[i]]推出,dp[i - 1][j - weight[i]] 为背包容量为j - weight[i]的时候不放物品i的最大价值,那么dp[i - 1][j - weight[i]] + value[i] (物品i的价值),就是背包放物品i得到的最大价值

所以递归公式: dp[i][j] = max(dp[i - 1][j], dp[i - 1][j - weight[i]] + value[i])。

如果改为滑动数组,则需倒序遍历背包容量,保证物品只被放进一次。

dp[j] = max(dp[j], dp[j - weight[i]] + value[i]);

分割等和子集

  • 背包的体积为sum / 2
  • 背包要放入的商品(集合里的元素)重量为 元素的数值,价值也为元素的数值
  • 背包如果正好装满,说明找到了总和为 sum / 2 的子集。
  • 背包中每一个元素是不可重复放入。

目标和

给定一个非负整数数组,a1, a2, ..., an, 和一个目标数,S。现在你有两个符号 + 和 -。对于数组中的任意一个整数,你都可以从 + 或 -中选择一个符号添加在前面。

返回可以使最终数组和为目标数 S 的所有添加符号的方法数。

假设加法的总和为x,那么减法对应的总和就是sum - x。

所以我们要求的是 x - (sum - x) = target

x = (target + sum) / 2

此时问题就转化为,装满容量为x的背包,有几种方法。求组合类问题的公式:

dp[j] += dp[j - nums[i]]

完全背包🎒

零钱兑换II


给定一个由正整数组成且不存在重复数字的数组,找出和为给定目标正整数的组合的个数。

排列总和 

给定一个由正整数组成且不存在重复数字的数组,找出和为给定目标正整数的排列的个数。

零钱兑换

给定不同面额的硬币 coins 和一个总金额 amount。编写一个函数来计算可以凑成总金额所需的最少的硬币个数。如果没有任何一种硬币组合能组成总金额,返回 -1。

完全平方数

完全平方数就是物品(可以无限件使用),凑个正整数n就是背包,问凑满这个背包最少有多少物品。

单词拆分

买卖股票

http://www.lryc.cn/news/413281.html

相关文章:

  • 回溯问题总结
  • GraphRAG如何使用ollama提供的llm model 和Embedding model服务构建本地知识库
  • .net # 检查 带有pdf xss
  • 【React】探讨className的正确使用方式
  • 打靶记录5——靶机hard_socnet2
  • 独立站+TikTok达人:自主营销与创意内容的完美结合
  • 【启明智显分享】适用于多功能养生壶、茶吧机的2.8寸触摸彩屏解决方案
  • WAF绕过技术(PKAV团队)
  • 『 Linux 』POSIX 信号量与基于环形队列的生产者消费者模型
  • python中的字符串方法
  • python实现consul的服务注册与注销
  • 校园选课助手【2】-重要的登录模块
  • 4章2节:从排序到分组和筛选,通过 R 的 dplyr 扩展包来操作
  • C语言实现 -- 单链表
  • WSL和Windows建立TCP通信协议
  • Android Gradle开发与应用(一):Gradle基础
  • Linux多线程服务器编程-1-线程安全的对象生命期管理
  • Couchbase 技术详解
  • PTE-信息收集
  • 委外订单执行明细表增加二开字段
  • “数字孪生+大模型“:打造设施农业全场景数字化运营新范式
  • zeppline 连接flink 1.17报错
  • 【机器视觉】【目标检测】【面试】独家问题总结表格
  • 从零开始,快速打造API:揭秘 Python 库toapi的神奇力量
  • 如何理解复信号z的傅里叶变换在频率v<0的时候恒为0,是解析信号
  • 大型赛事5G室内无线网络保障方案
  • windows 2012域服务SYSVOL复制异常
  • 动态规划,蒙特卡洛,TD,Qlearing,Sars,DQN,REINFORCE算法对比
  • HarmonyOS开发商城商品详情页
  • OS_操作系统的运行环境