LeetCode最长序列

简介: LeetCode最长序列

最长序列


300. 最长递增子序列【中等】



  • 方法一:动态规划


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


空间复杂度:O(n)


1、确定dp数组及其含义


  • dp[i]表示0~i索引的最长上升子序列长度


2、确定状态转移方程


  • if(nums[i] > nums[j]) dp[i] = max(dp[i], dp[j]+1);


3、dp初始化


  • dp[i] = 1;,只要nums有长度,至少为1


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


方法二:贪心+二分查找


时间复杂度:O(nlogn)


空间复杂度:O(n)


674. 最长连续递增序列【简单】



思路一:动态规划


时间复杂度:O(n)


空间复杂度:O(n)


1、确定dp数组及其下标含义


  • dp[i]是索引0~i时连续递增子序列长度


2、确定递推公式


  • if(nums[i] > nums[i-1]) dp[i] = dp[i-1] + 1;


3、dp初始化


  • vector<int> dp(nums.size(), 1);


class Solution {
public:
    int findLengthOfLCIS(vector<int>& nums) {
        if (nums.size() <= 1) return nums.size();
        int maxLen = 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;
            maxLen = max(maxLen, dp[i]);
        }
        return maxLen;
    }
};


  • 思路二:贪心


时间复杂度:O(n)


空间复杂度:O(1)


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


思路三:双指针


时间复杂度:O(n)


空间复杂度:O(1)


class Solution {
public:
    int findLengthOfLCIS(vector<int>& nums) {
        if (nums.size() <= 1) return nums.size();
        int maxLen = 1;
    int start = 0;
        for(int end = 1; end < nums.size(); ++end){
            if(nums[end] <= nums[end-1]){
                maxLen = max(maxLen, end - start);
                start = end;
            }
        }
        //最后一节
        if(nums.size() - start > maxLen) maxLen = nums.size() - start;
        return maxLen;
    }
};


718. 最长重复子数组【中等】



思路:动态规划


时间复杂度:O(mn)


空间复杂度:O(mn)


1、确定dp数组及其下标含义


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


2、确定递推公式


  • dp[i][j] = dp[i-1][j-1] + 1;


3、dp初始化


  • dp[i][0] = 0;


  • dp[0][j] = 0;


class Solution {
public:
    int findLength(vector<int>& A, vector<int>& B) {
        vector<vector<int>> dp (A.size() + 1, vector<int>(B.size() + 1, 0));
        int maxLen = 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;
                }
                maxLen = max(maxLen, dp[i][j]);
            }
        }
        return maxLen;
    }
};


思路二:空间优化,滚动数组


class Solution {
public:
    int findLength(vector<int>& A, vector<int>& B) {
        vector<int> dp(vector<int>(B.size() + 1, 0));
        int maxLen = 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的操作
                maxLen = max(maxLen, dp[j]);
            }
        }
        return maxLen;
    }
};


1143. 最长公共子序列【中等】




  • 思路:动态规划


时间复杂度:O(mn)


空间复杂度:O(mn)


1、确定dp数组以及下标含义


  • dp[i][j]:索引为0-i的text1与0-j的text2的最长公共子序列


2、确定递推公式


  • if(text1[i-1] == text2[j-1]) dp[i][j] = dp[i-1][j-1] + 1;


  • if(text1[i-1] != text2[j-1]) dp[i][j] != max(dp[i-1][j], dp[i][j-1]);


3、dp初始化


  • vector<vector<int>> dp(text1.size(), vector<int>(text2.size(), 0));


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()];
    }
};
目录
相关文章
|
5月前
|
Python
【Leetcode刷题Python】376. 摆动序列
文章提供了解决LeetCode "摆动序列" 问题的Python实现代码,通过遍历整数数组并使用两个变量 down 和 up 来记录正差和负差摆动序列的长度,最终返回最长摆动子序列的长度。
45 0
|
8月前
|
存储 算法
《LeetCode》—— 摆动序列
《LeetCode》—— 摆动序列
|
8月前
|
算法 安全 测试技术
[组合数学]LeetCode:2954:统计感冒序列的数目
[组合数学]LeetCode:2954:统计感冒序列的数目
|
8月前
|
算法 测试技术 C#
【单调栈】LeetCode2030:含特定字母的最小子序列
【单调栈】LeetCode2030:含特定字母的最小子序列
|
5月前
|
Python
【Leetcode刷题Python】946. 验证栈序列
LeetCode题目“946. 验证栈序列”的Python解决方案,通过模拟栈的压入和弹出操作来验证给定的两个序列是否能通过合法的栈操作得到。
37 6
|
5月前
|
算法 Python
【Leetcode刷题Python】剑指 Offer 33. 二叉搜索树的后序遍历序列
本文提供了一种Python算法,用以判断给定整数数组是否为某二叉搜索树的后序遍历结果,通过识别根节点并递归验证左右子树的值是否满足二叉搜索树的性质。
28 3
|
5月前
|
Python
【Leetcode刷题Python】105. 从前序与中序遍历序列构造二叉树
LeetCode上105号问题"从前序与中序遍历序列构造二叉树"的Python实现,通过递归方法根据前序和中序遍历序列重建二叉树。
33 3
|
5月前
|
算法 Python
【Leetcode刷题Python】300. 最长递增子序列
LeetCode 300题 "最长递增子序列" 的两种Python解决方案:一种使用动态规划,另一种使用贪心算法结合二分查找。
44 1
|
5月前
|
算法 Java
LeetCode初级算法题:子数组最大平均数+二叉树的最小深度+最长连续递增序列+柠檬水找零
LeetCode初级算法题:子数组最大平均数+二叉树的最小深度+最长连续递增序列+柠檬水找零
45 0
|
5月前
|
Python
【Leetcode刷题Python】674. 最长连续递增序列
LeetCode 674题 "最长连续递增序列" 的Python解决方案,使用动态规划算法找出给定整数数组中最长连续递增子序列的长度。
111 0