子序列问题-代码随想录-刷题笔记

子序列问题-代码随想录-刷题笔记_第1张图片

300. 最长递增子序列-有序递增

1)dp[i]:表示i之前包括i的以nums[i]结尾的最长递增子序列的长度
为什么一定表示 “以nums[i]结尾的最长递增子序” ,因为我们在做递增比较的时候,如果比较 nums[j] 和 nums[i] 的大小,那么两个递增子序列一定分别以nums[j]为结尾 和 nums[i]为结尾, 要不然这个比较就没有意义了,不是尾部元素的比较那么 如何算递增呢。
2)递推公式
位置i的最长升序子序列等于j从0到i-1各个位置的最长升序子序列 + 1 的最大值。
所以:if (nums[i] > nums[j]) dp[i] = max(dp[i], dp[j] + 1);
注意这里不是要dp[i] 与 dp[j] + 1进行比较,而是我们要取dp[j] + 1的最大值
3)初始化
dp[0]=1
4)方向
从前到后
5)举例
子序列问题-代码随想录-刷题笔记_第2张图片

整体代码

//时间复杂度: O(n^2)
//空间复杂度: O(n)
class Solution {
public:
    int lengthOfLIS(vector<int>& nums) {
        if (nums.size() <= 1) return nums.size();
        vector<int> dp(nums.size(), 1);
        int result = 0;
        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);
            }
            if (dp[i] > result) result = dp[i]; // 取长的子序列
        }
        return result;
    }
};

674. 最长连续递增序列-连续有序递增

动态规划

1)dp[i]:以下标i为结尾的连续递增的子序列长度为dp[i]
2)递推公式:if(nums[i]>nums[i-1]) dp[i]=dp[i-1]+1
3)初始化 dp[i]=1
4)方向:从前到后
5)举例
子序列问题-代码随想录-刷题笔记_第3张图片
整体代码

//时间复杂度: O(n)
//空间复杂度: O(n)
class Solution {
public:
    int findLengthOfLCIS(vector<int>& nums) {
        if (nums.size() == 0) return 0;
        int result = 1;
        vector<int> dp(nums.size() ,1);
        for (int i = 1; i < nums.size(); i++) {
            if (nums[i] > nums[i - 1]) { // 连续记录
                dp[i] = dp[i - 1] + 1;
            }
            if (dp[i] > result) result = dp[i];
        }
        return result;
    }
};

贪心

//时间复杂度: O(n)
//空间复杂度: O(1)
class Solution {
public:
    int findLengthOfLCIS(vector<int>& nums) {
        if (nums.size() == 0) return 0;
        int result = 1; // 连续子序列最少也是1
        int count = 1;
        for (int i = 1; i < nums.size(); i++) {
            if (nums[i] > nums[i - 1]) { // 连续记录
                count++;
            } else { // 不连续,count从头开始
                count = 1;
            }
            if (count > result) result = count;
        }
        return result;
    }
};

718. 最长重复子数组-连续有序

1)dp[i][j]含义
dp[i][j] :以下标i - 1为结尾的A,和以下标j - 1为结尾的B,最长重复子数组长度为dp[i][j]。 (特别注意: “以下标i - 1为结尾的A” 标明一定是 以A[i-1]为结尾的字符串 )
其实dp[i][j]的定义也就决定着,我们在遍历dp[i][j]的时候i 和 j都要从1开始。
2)递推公式
当A[i - 1] 和B[j - 1]相等的时候,dp[i][j] = dp[i - 1][j - 1] + 1;
3)初始化:
dp[i][0] 和dp[0][j]初始化为0
4)方向:
外层for循环遍历A,内层for循环遍历B,反过来也一样
5)举例
子序列问题-代码随想录-刷题笔记_第4张图片

整体代码

// 版本一
//时间复杂度:O(n × m),n 为A长度,m为B长度
//空间复杂度:O(n × m)
class Solution {
public:
    int findLength(vector<int>& nums1, vector<int>& nums2) {
        vector<vector<int>> dp (nums1.size() + 1, vector<int>(nums2.size() + 1, 0));
        int result = 0;
        for (int i = 1; i <= nums1.size(); i++) {
            for (int j = 1; j <= nums2.size(); j++) {
                if (nums1[i - 1] == nums2[j - 1]) {
                    dp[i][j] = dp[i - 1][j - 1] + 1;
                }
                if (dp[i][j] > result) result = dp[i][j];
            }
        }
        return result;
    }
};

滚动数组

子序列问题-代码随想录-刷题笔记_第5张图片
可以看出dp[i][j]都是由dp[i - 1][j - 1]推出。那么压缩为一维数组,也就是dp[j]都是由dp[j - 1]推出。
也就是相当于可以把上一层dp[i - 1][j]拷贝到下一层dp[i][j]来继续用
此时遍历B数组的时候,就要从后向前遍历,这样避免重复覆盖

// 版本二
//时间复杂度:O(n × m),n 为A长度,m为B长度
//空间复杂度:O(m)
class Solution {
public:
    int findLength(vector<int>& A, vector<int>& B) {
        vector<int> dp(vector<int>(B.size() + 1, 0));
        int result = 0;
        for (int i = 1; i <= A.size(); i++) {
            for (int j = B.size(); j > 0; j--) {
                if (A[i - 1] == B[j - 1]) {
                    dp[j] = dp[j - 1] + 1;
                } else dp[j] = 0; // 注意这里不相等的时候要有赋0的操作
                if (dp[j] > result) result = dp[j];
            }
        }
        return result;
    }
};

1143.最长公共子序列-不连续有序

1)dp[i][j]:以下标i-1结束的text1和以下标j-1结束的text2的最长公共子序列长度为dp[i][j]
2)递推公式:
当A[i - 1] 和B[j - 1]相等的时候,dp[i][j] = dp[i - 1][j - 1] + 1;
不相等的时候dp[i][j] = max(dp[i - 1][j ],dp[i][j-1]);
3)初始化:
dp[i][0] 和dp[0][j]初始化为0
4)方向:从上到下,从左到右
子序列问题-代码随想录-刷题笔记_第6张图片
5)举例
子序列问题-代码随想录-刷题笔记_第7张图片

整体代码

//时间复杂度: O(n * m),其中 n 和 m 分别为 text1 和 text2 的长度
//空间复杂度: O(n * m)
class Solution {
public:
    int longestCommonSubsequence(string text1, string text2) {
        vector<vector<int>> dp(text1.size() + 1, vector<int>(text2.size() + 1, 0));
        for (int i = 1; i <= text1.size(); i++) {
            for (int j = 1; j <= text2.size(); 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[text1.size()][text2.size()];
    }
};

1035.不相交的线:就是最长公共子序列

求绘制的最大连线数,其实就是求两个字符串的最长公共子序列的长度!
1)dp[i][j]:以下标i-1结束的text1和以下标j-1结束的text2的最长公共子序列长度为dp[i][j]
2)递推公式:
当A[i - 1] 和B[j - 1]相等的时候,dp[i][j] = dp[i - 1][j - 1] + 1;
不相等的时候dp[i][j] = max(dp[i - 1][j ],dp[i][j-1]);
3)初始化:
dp[i][0] 和dp[0][j]初始化为0
4)方向:从上到下,从左到右

整体代码

//时间复杂度: O(n * m)
//空间复杂度: O(n * m)
class Solution {
public:
    int maxUncrossedLines(vector<int>& A, vector<int>& B) {
        vector<vector<int>> dp(A.size() + 1, vector<int>(B.size() + 1, 0));
        for (int i = 1; i <= A.size(); i++) {
            for (int j = 1; j <= B.size(); j++) {
                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]);
                }
            }
        }
        return dp[A.size()][B.size()];
    }
};

53. 最大子序和

贪心

//时间复杂度: O(n)
//空间复杂度: O(1)
class Solution {
public:
    int maxSubArray(vector<int>& nums) {   
        int ans=INT_MIN;
        int sum=0;
        for(int i=0;i<nums.size();i++){
           sum+=nums[i];
           if(sum<nums[i]){
               sum=nums[i];
           }
           ans=max(ans,sum);
        }
        return ans;
    }
};

动态规划

1)dp[i]:包括下标i(以nums[i]为结尾)的最大连续子序列和为dp[i]
2)递归公式:
dp[i]=max(dp[i-1]+nums[i],nums[i])
3)初始化:
dp[0]=nums[0]
4 )方向
从前到后
5)举例
子序列问题-代码随想录-刷题笔记_第8张图片

整体代码

//时间复杂度: O(n)
//空间复杂度: O(n)
class Solution {
public:
    int maxSubArray(vector<int>& nums) {
        if (nums.size() == 0) return 0;
        vector<int> dp(nums.size());
        dp[0] = nums[0];
        int result = dp[0];
        for (int i = 1; i < nums.size(); i++) {
            dp[i] = max(dp[i - 1] + nums[i], nums[i]); // 状态转移公式
            if (dp[i] > result) result = dp[i]; // result 保存dp[i]的最大值
        }
        return result;
    }
};

392.判断子序列

双指针

//时间复杂度: O(n)
//空间复杂度: O(1)
class Solution {
public:
    bool isSubsequence(string s, string t) {
       int i=0,j=0;
       while(i<s.size()&&j<t.size()){
           if(s[i]==t[j]){
               i++;
               j++;
           }
           else{
               j++;
           }
       }
       return i==s.size();
    }
};

动态规划

1)dp[i][j] :以下标i-1为结尾的字符串s,和以下标j-1为结尾的字符串t,相同子序列的长度为dp[i][j]
2)递推公式:

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

3)初始化:
子序列问题-代码随想录-刷题笔记_第9张图片
4)顺序:从上到下,从左到右
子序列问题-代码随想录-刷题笔记_第10张图片
5)举例
子序列问题-代码随想录-刷题笔记_第11张图片

整体代码

//时间复杂度:O(n × m)
//空间复杂度:O(n × m)
class Solution {
public:
    bool isSubsequence(string s, string t) {
       vector<vector<int>>dp(s.size()+1,vector<int>(t.size()+1,0));
       for(int i=1;i<=s.size();i++){
           for(int j=1;j<=t.size();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[s.size()][t.size()]==s.size();
    }
};

115.不同的子序列

  1. 确定dp数组(dp table)以及下标的含义

dp[i][j]:以i-1为结尾的s子序列中出现以j-1为结尾的t的个数为dp[i][j]
2.确定递推公式

  • s[i - 1] 与 t[j - 1]相等
  • s[i - 1] 与 t[j - 1] 不相等

当s[i - 1] 与 t[j - 1]相等时:dp[i][j] = dp[i - 1][j - 1] + dp[i - 1][j];
一部分是用s[i - 1]来匹配,那么个数为dp[i - 1][j - 1]。即不需要考虑当前s子串和t子串的最后一位字母,所以只需要 dp[i-1][j-1]。
一部分是不用s[i - 1]来匹配,个数为dp[i - 1][j]
为什么还要考虑 不用s[i - 1]来匹配,都相同了指定要匹配啊
例如: s:bagg 和 t:bag ,s[3] 和 t[2]是相同的,但是字符串s也可以不用s[3]来匹配,即用s[0]s[1]s[2]组成的bag。
当然也可以用s[3]来匹配,即:s[0]s[1]s[3]组成的bag
当s[i - 1] 与 t[j - 1]不相等时:dp[i][j] = dp[i - 1][j];
dp[i][j]只有一部分组成,不用s[i - 1]来匹配(就是模拟在s中删除这个元素),即:dp[i - 1][j]
为什么只考虑 “不用s[i - 1]来匹配” 这种情况, 不考虑 “不用t[j - 1]来匹配” 的情况呢。
我们求的是 s 中有多少个 t,而不是 求t中有多少个s,所以只考虑 s中删除元素的情况,即 不用s[i - 1]来匹配 的情况
3.dp数组如何初始化
从递推公式中可以看出dp[i][j] 是从上方和左上方推导而来,如图:,那么 dp[i][0] 和dp[0][j]是一定要初始化的。
子序列问题-代码随想录-刷题笔记_第12张图片
dp[i][0] 表示:以i-1为结尾的s可以随便删除元素,出现空字符串的个数。
那么dp[i][0]一定都是1,因为也就是把以i-1为结尾的s,删除所有元素,出现空字符串的个数就是1。
再来看dp[0][j],dp[0][j]:空字符串s可以随便删除元素,出现以j-1为结尾的字符串t的个数。
那么dp[0][j]一定都是0,s如论如何也变成不了t。
最后就要看一个特殊位置了,即:dp[0][0] 应该是多少。
dp[0][0]应该是1,空字符串s,可以删除0个元素,变成空字符串t
4.确定遍历顺序
从上到下,从左到右
5.举例推导dp数组
子序列问题-代码随想录-刷题笔记_第13张图片

整体代码

//时间复杂度: O(n * m)
//空间复杂度: O(n * m)
class Solution {
public:
    int numDistinct(string s, string t) {
        vector<vector<uint64_t>> dp(s.size() + 1, vector<uint64_t>(t.size() + 1));
        for (int i = 0; i < s.size(); i++) dp[i][0] = 1;
        for (int j = 1; j < t.size(); j++) dp[0][j] = 0;
        for (int i = 1; i <= s.size(); i++) {
            for (int j = 1; j <= t.size(); j++) {
                if (s[i - 1] == t[j - 1]) {
                    dp[i][j] = dp[i - 1][j - 1] + dp[i - 1][j];
                } else {
                    dp[i][j] = dp[i - 1][j];
                }
            }
        }
        return dp[s.size()][t.size()];
    }
};

583. 两个字符串的删除操作

  1. 确定dp数组(dp table)以及下标的含义

dp[i][j]:以i-1为结尾的字符串word1,和以j-1位结尾的字符串word2,想要达到相等,所需要删除元素的最少次数。
2.确定递推公式

  • 当word1[i - 1] 与 word2[j - 1]相同的时候
  • 当word1[i - 1] 与 word2[j - 1]不相同的时候

当word1[i - 1] 与 word2[j - 1]相同的时候,dp[i][j] = dp[i - 1][j - 1];
当word1[i - 1] 与 word2[j - 1]不相同的时候,有三种情况:
情况一:删word1[i - 1],最少操作次数为dp[i - 1][j] + 1
情况二:删word2[j - 1],最少操作次数为dp[i][j - 1] + 1
情况三:同时删word1[i - 1]和word2[j - 1],操作的最少次数为dp[i - 1][j - 1] + 2
那最后当然是取最小值,所以当word1[i - 1] 与 word2[j - 1]不相同的时候,递推公式:dp[i][j] = min({dp[i - 1][j - 1] + 2, dp[i - 1][j] + 1, dp[i][j - 1] + 1});
因为 dp[i][j - 1] + 1 = dp[i - 1][j - 1] + 2,所以递推公式可简化为:dp[i][j] = min(dp[i - 1][j] + 1, dp[i][j - 1] + 1);从字面上理解 就是 当 同时删word1[i - 1]和word2[j - 1],dp[i][j-1] 本来就不考虑 word2[j - 1]了,那么我在删 word1[i - 1],是不是就达到两个元素都删除的效果,即 dp[i][j-1] + 1。
3.dp数组如何初始化
dp[i][0]:word2为空字符串,以i-1为结尾的字符串word1要删除多少个元素,才能和word2相同呢,很明显dp[i][0] = i。
dp[0][j]的话同理
4.确定遍历顺序
从上到下,从左到右
5.举例
子序列问题-代码随想录-刷题笔记_第14张图片

整体代码

动态规划一

//时间复杂度: O(n * m)
//空间复杂度: O(n * m)
class Solution {
public:
    int minDistance(string word1, string word2) {
        vector<vector<int>> dp(word1.size() + 1, vector<int>(word2.size() + 1));
        for (int i = 0; i <= word1.size(); i++) dp[i][0] = i;
        for (int j = 0; j <= word2.size(); j++) dp[0][j] = j;
        for (int i = 1; i <= word1.size(); i++) {
            for (int j = 1; j <= word2.size(); 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] + 1, dp[i][j - 1] + 1);
                }
            }
        }
        return dp[word1.size()][word2.size()];
    }
};

动态规划二

本题和动态规划:1143.最长公共子序列(opens new window)基本相同,只要求出两个字符串的最长公共子序列长度即可,那么除了最长公共子序列之外的字符都是必须删除的,最后用两个字符串的总长度减去两个最长公共子序列的长度就是删除的最少步数。

//时间复杂度: O(n * m)
//空间复杂度: O(n * m)
class Solution {
public:
    int minDistance(string word1, string word2) {
        vector<vector<int>> dp(word1.size()+1, vector<int>(word2.size()+1, 0));
        for (int i=1; i<=word1.size(); i++){
            for (int j=1; j<=word2.size(); j++){
                if (word1[i-1] == word2[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 word1.size()+word2.size()-dp[word1.size()][word2.size()]*2;
    }
};

72. 编辑距离

1. 确定dp数组(dp table)以及下标的含义

dp[i][j] 表示以下标i-1为结尾的字符串word1,和以下标j-1为结尾的字符串word2,最近编辑距离为dp[i][j]

2. 确定递推公式

if (word1[i - 1] == word2[j - 1])
    不操作
    dp[i][j] = dp[i - 1][j - 1];
if (word1[i - 1] != word2[j - 1])
    增
    删
    换

操作一:word1删除一个元素,那么就是以下标i - 2为结尾的word1 与 j-1为结尾的word2的最近编辑距离 再加上一个操作:dp[i][j] = dp[i - 1][j] + 1
操作二:word2删除一个元素,那么就是以下标i - 1为结尾的word1 与 j-2为结尾的word2的最近编辑距离 再加上一个操作:dp[i][j] = dp[i][j - 1] + 1
word2添加一个元素,相当于word1删除一个元素,例如 word1 = “ad” ,word2 = “a”,word1删除元素’d’ 和 word2添加一个元素’d’,变成word1=“a”, word2=“ad”, 最终的操作数是一样! dp数组如下图
子序列问题-代码随想录-刷题笔记_第15张图片
操作三:替换元素,word1替换word1[i - 1],使其与word2[j - 1]相同,此时不用增删加元素。可以回顾一下,if (word1[i - 1] == word2[j - 1])的时候我们的操作 是 dp[i][j] = dp[i - 1][j - 1] 对吧。那么只需要一次替换的操作,就可以让 word1[i - 1] 和 word2[j - 1] 相同。所以 dp[i][j] = dp[i - 1][j - 1] + 1;

当 if (word1[i - 1] != word2[j - 1]) 时取最小的,即:dp[i][j] = min({dp[i - 1][j - 1], dp[i - 1][j], dp[i][j - 1]}) + 1;

3. dp数组如何初始化

dp[i][0] :以下标i-1为结尾的字符串word1,和空字符串word2,最近编辑距离为dp[i][0]
dp[i][0]就应该是i,对word1里的元素全部做删除操作,即:dp[i][0] = i;
同理dp[0][j] = j;

4. 确定遍历顺序

从上到下,从左到右

子序列问题-代码随想录-刷题笔记_第16张图片

5.举例推导dp数组

子序列问题-代码随想录-刷题笔记_第17张图片

整体代码

//时间复杂度: O(n * m)
//空间复杂度: O(n * m)
class Solution {
public:
    int minDistance(string word1, string word2) {
        vector<vector<int>> dp(word1.size() + 1, vector<int>(word2.size() + 1, 0));
        for (int i = 0; i <= word1.size(); i++) dp[i][0] = i;
        for (int j = 0; j <= word2.size(); j++) dp[0][j] = j;
        for (int i = 1; i <= word1.size(); i++) {
            for (int j = 1; j <= word2.size(); 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 - 1], dp[i - 1][j], dp[i][j - 1]}) + 1;
                }
            }
        }
        return dp[word1.size()][word2.size()];
    }
};

  1. 回文子串
    子序列问题-代码随想录-刷题笔记_第18张图片

  2. 确定dp数组(dp table)以及下标的含义

我们找到一种递归关系,也就是判断一个子字符串(字符串的下表范围[i,j])是否回文,依赖于,子字符串(下表范围[i + 1, j - 1])) 是否是回文。
为了明确这种递归关系,我们的dp数组是要定义成二维dp数组。
布尔类型的dp[i][j]:表示区间范围[i,j] (注意是左闭右闭)的子串是否是回文子串,如果是dp[i][j]为true,否则为false
2.确定递推公式
当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
//result就是统计回文子串的数量。
if (s[i] == s[j]) {
    if (j - i <= 1) { // 情况一 和 情况二
        result++;
        dp[i][j] = true;
    } else if (dp[i + 1][j - 1]) { // 情况三
        result++;
        dp[i][j] = true;
    }
}

3.dp数组如何初始化
dp[i][j]可以初始化为true么? 当然不行,怎能刚开始就全都匹配上了。
所以dp[i][j]初始化为false
4.确定遍历顺序
情况三是根据dp[i + 1][j - 1]是否为true,再对dp[i][j]进行赋值true的
子序列问题-代码随想录-刷题笔记_第19张图片
一定要从下到上,从左到右遍历,这样保证dp[i + 1][j - 1]都是经过计算的
5.举例推导dp数组
输入:“aaa”,dp[i][j]状态如下:
子序列问题-代码随想录-刷题笔记_第20张图片

完整代码

//时间复杂度:O(n^2)
//空间复杂度:O(n^2)
class Solution {
public:
    int countSubstrings(string s) {
        vector<vector<bool>> dp(s.size(), vector<bool>(s.size(), false));
        int result = 0;
        for (int i = s.size() - 1; i >= 0; i--) {  // 注意遍历顺序
            for (int j = i; j < s.size(); j++) {
                if (s[i] == s[j]) {
                    if (j - i <= 1) { // 情况一 和 情况二
                        result++;
                        dp[i][j] = true;
                    } else if (dp[i + 1][j - 1]) { // 情况三
                        result++;
                        dp[i][j] = true;
                    }
                }
            }
        }
        return result;
    }
};

简化一下

//时间复杂度:O(n^2)
//空间复杂度:O(n^2)
class Solution {
public:
    int countSubstrings(string s) {
        vector<vector<bool>> dp(s.size(), vector<bool>(s.size(), false));
        int result = 0;
        for (int i = s.size() - 1; i >= 0; i--) {
            for (int j = i; j < s.size(); j++) {
                if (s[i] == s[j] && (j - i <= 1 || dp[i + 1][j - 1])) {
                    result++;
                    dp[i][j] = true;
                }
            }
        }
        return result;
    }
};

双指针法:中心扩散法

//时间复杂度:O(n^2)
//空间复杂度:O(1)
class Solution {
public:
    int countSubstrings(string s) {
        int result = 0;
        for (int i = 0; i < s.size(); i++) {
            result += extend(s, i, i, s.size()); // 以i为中心
            result += extend(s, i, i + 1, s.size()); // 以i和i+1为中心
        }
        return result;
    }
    int extend(const string& s, int i, int j, int n) {
        int res = 0;
        while (i >= 0 && j < n && s[i] == s[j]) {
            i--;
            j++;
            res++;
        }
        return res;
    }
};

516.最长回文子序列

  1. 确定dp数组(dp table)以及下标的含义

dp[i][j]:字符串s在[i, j]范围内最长的回文子序列的长度为dp[i][j]
2.确定递推公式
如果s[i]与s[j]相同,那么dp[i][j] = dp[i + 1][j - 1] + 2;
子序列问题-代码随想录-刷题笔记_第21张图片
如果s[i]与s[j]不相同,说明s[i]和s[j]的同时加入 并不能增加[i,j]区间回文子序列的长度,那么分别加入s[i]、s[j]看看哪一个可以组成最长的回文子序列。
看图:加入s[j]的回文子序列长度为dp[i + 1][j]。
看图:加入s[i]的回文子序列长度为dp[i][j - 1]。
那么dp[i][j]一定是取最大的,即:dp[i][j] = max(dp[i + 1][j], dp[i][j - 1]);
子序列问题-代码随想录-刷题笔记_第22张图片
3.dp数组如何初始化
需要手动初始化一下,当i与j相同,那么dp[i][j]一定是等于1的,即:一个字符的回文子序列长度就是1。
其他情况dp[i][j]初始为0就行,这样递推公式:dp[i][j] = max(dp[i + 1][j], dp[i][j - 1]); 中dp[i][j]才不会被初始值覆盖。

vector<vector<int>> dp(s.size(), vector<int>(s.size(), 0));
for (int i = 0; i < s.size(); i++) dp[i][i] = 1;

4.确定遍历顺序
子序列问题-代码随想录-刷题笔记_第23张图片
所以遍历i的时候一定要从下到上遍历,这样才能保证下一行的数据是经过计算的
j的话,可以正常从左向右遍历。
5.举例推导dp数组
红色框即:dp[0][s.size() - 1]; 为最终结果。
子序列问题-代码随想录-刷题笔记_第24张图片

整体代码

//时间复杂度: O(n^2)
//空间复杂度: O(n^2)
class Solution {
public:
    int longestPalindromeSubseq(string s) {
        vector<vector<int>> dp(s.size(), vector<int>(s.size(), 0));
        for (int i = 0; i < s.size(); i++) dp[i][i] = 1;
        for (int i = s.size() - 1; i >= 0; i--) {
            for (int j = i + 1; j < s.size(); 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]);
                }
            }
        }
        return dp[0][s.size() - 1];
    }
};

你可能感兴趣的:(笔记,leetcode,算法,c++,数据结构,动态规划)