[LeetCode]300. 最长递增子序列

75 篇文章 0 订阅

题目

300. 最长递增子序列
给你一个整数数组 nums ,找到其中最长严格递增子序列的长度。

子序列 是由数组派生而来的序列,删除(或不删除)数组中的元素而不改变其余元素的顺序。例如,[3,6,2,7] 是数组 [0,3,1,6,2,2,7] 的子序列。

 
示例 1:

输入:nums = [10,9,2,5,3,7,101,18]
输出:4
解释:最长递增子序列是 [2,3,7,101],因此长度为 4 。
示例 2:

输入:nums = [0,1,0,3,2,3]
输出:4
示例 3:

输入:nums = [7,7,7,7,7,7,7]
输出:1
 

提示:

1 <= nums.length <= 2500
-104 <= nums[i] <= 104
 

进阶:

你能将算法的时间复杂度降低到 O(n log(n))?

一些名词

  • 最长上升子序列( L I S LIS LIS):Longest Increasing Subsequence
  • 最长连续序列( L C S LCS LCS):Longest Consecutive Sequence
  • 最长连续递增序列( L C I S LCIS LCIS):Longest Continuous Increasing Subsequence
  • 最长公共子序列( L C S LCS LCS):Longest Common Subsequence

子串与子序列区别:子串不可跳跃,子序列可以跳跃,如 “AC”是“ABCDEFG”的子序列,而不是子串。 而“ABC”则是其子串

1589761403470.png

定义状态

  • d p [ i ] dp[i] dp[i]表示在区间 n u m s [ 0.... i ] nums[0....i] nums[0....i]区间范围内的最长上升子序列的长度

2020-05-18_082043.png

  • 比较索引 i i i与其前面出现的某个位置 j j j的大小
    • n u m s [ i ] < = n u m s [ j ] nums[i]<=nums[j] nums[i]<=nums[j],说明 j j j处的值要比 i i i处的值的,要是形成子序列则是 n u m s [ 0... j , i ] nums[0...j,i] nums[0...j,i]这样的结果,这时候 j j j i i i不能形成递增, i i i结尾的子序列所形成的最长子序列的长度等价于以 j j j结尾的子序列所形成的最长子序列的长度,即 d p [ i ] = d p [ j ] dp[i]=dp[j] dp[i]=dp[j]
    • n u m s [ i ] > n u m s [ j ] nums[i]>nums[j] nums[i]>nums[j],说明 j j j处的值小于 i i i处值,形成的子序列是 n u m s [ 0... j , i ] nums[0...j,i] nums[0...j,i]这样的结果,这时候从 j j j i i i是递增的,这时候需要在长度上+1,即 d p [ i ] = d p [ j + 1 ] dp[i]=dp[j+1] dp[i]=dp[j+1]
    • 取上述两种情况的 m a x max max,动态转移方程为: d p [ i ] = M a t h . m a x ( d p [ i ] , d p [ j ] + 1 ) ∣ 0 < = j < i < n dp[i] = Math.max(dp[i], dp[j] + 1)|0<=j<i<n dp[i]=Math.max(dp[i],dp[j]+1)0<=j<i<n
    • 举例:如果遍历到 i i i位置,在 [ 0 − i ] [0-i] [0i] 区间内有 [ 0 − j ] j < i [0-j] j<i [0j]j<i n u m s [ i ] < = n u m s [ j ] nums[i]<=nums[j] nums[i]<=nums[j]时,表示以 j j j结束的子序列和 i i i结束的子序列不能形成上升子序列,举 例: [ 1 , 4 , 5 , 7 , 6 , 8 ] [1,4,5,7,6,8] [1,4,5,7,6,8],当 i i i i n d e x index index 4 4 4的位置,也就是 n u m s [ i ] = 6 nums[i] =6 nums[i]=6 ,$j 为 为 index$ 为 3 3 3时, n u m s [ j ] = 7 nums[j] =7 nums[j]=7 ,以$nums[j] 和 和 nums[i]$ 不能形成一个上升子序列

边界条件

  • n u m s [ 0... i ] nums[0...i] nums[0...i]只有一个元素时,即以这一个元素结尾的子序列,最长上升子序列是其自身,为 1 1 1

核心代码

 for (int i = 0; i < len; i++) {
            for (int j = 0; j < i; j++) {
                if (nums[i] > nums[j]) {
                    dp[i] = Math.max(dp[i], dp[j] + 1);
                }
            }
        }

方法1:DP

 public int lengthOfLIS(int[] nums) {
        //dp[i]: 到i为止 (对于所有 j in [0, i], 记录max length of increasing subsequence
        if (nums == null || nums.length == 0) {
            return 0;
        }
        int len = nums.length;
        int[] dp = new int[len];
        int res = 0;
        for (int i = 0; i < len; i++) {
            dp[i] = 1;
            for (int j = 0; j < i; j++) {
                //i 位置的数与[0,i]位置之间的数比较,如果大于进逻辑
                if (nums[i] > nums[j]) {
                    //等于dp[i]或者dp[j] + 1(j对应的值比i小)的最大值
                    dp[i] = Math.max(dp[i], dp[j] + 1);
                }
            }
            res = Math.max(res, dp[i]);
        }
        return res;
    }

方法2:贪心+二分

  • 准备一个辅助数组 t a i l s tails tails,其中 t a i l s [ i ] tails[i] tails[i]表示长度为 i + 1 i+1 i+1 n u m s [ 0... i ] nums[0...i] nums[0...i]的序列尾部元素的值
  • 辅助数组 t a i l s tails tails一定是严格单调递增的,即在 n u m s [ 0... j . . i ] nums[0...j..i] nums[0...j..i]区间上, t a i l s [ j ] < t a i l s [ i ] tails[j]<tails[i] tails[j]<tails[i],下面使用 反证法证明这个结论
    • 假设 n u m s [ 0... j . . i ] nums[0...j..i] nums[0...j..i]这个区间上, t a i l s [ j ] > = t a i l s [ i ] tails[j]>=tails[i] tails[j]>=tails[i],从 i i i这个子序列向前删除 i − j i-j ij个元素,这时候长度变为 j j j的子序列,这时候的尾部元素的值为 x x x
    • 根据 t a i l s tails tails数组的定义, x < t a i l s [ i ] x<tails[i] x<tails[i]
    • x x x又是 t a i l s [ j ] tails[j] tails[j]的值,即 x = t a i l s [ j ] x=tails[j] x=tails[j]
    • 得出 t a i l s [ i ] > t a i l s [ j ] tails[i]>tails[j] tails[i]>tails[j],这与一开始假设的条件矛盾,假设条件不成立
 public int lengthOfLISII(int[] nums) {
        int n = nums.length;
        int[] tails = new int[n];
        int end = 0;
        for (int i = 0; i < n; i++) {
            int l = 0, r = end;
            while (l < r) {
                int m = (l + r) / 2;
                if (tails[m] < nums[i]) l = m + 1;
                else r = m;
            }
            tails[l] = nums[i];
            if (end == r) end++;
        }
        return end;
    }
  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值