Given an unsorted array of integers, find the length of longest increasing subsequence.

For example,
Given [10, 9, 2, 5, 3, 7, 101, 18],
The longest increasing subsequence is [2, 3, 7, 101], therefore the length is 4. Note that there may be more than one LIS combination, it is only necessary for you to return the length.

Your algorithm should run in O(n2) complexity.

Follow up: Could you improve it to O(n log n) time complexity?

思路:DP。这题有两种解法。

首先,是经典的O(N^2)解法。

我们维护一个一维数组,dp[i]表示以nums[i]为最后一位的最长递增子序列的长度。

递推公式为dp[i] = max(dp[j] + 1) 其中 0 <= j < i 且 nums[j] < nums[i]。

 class Solution {
public:
int lengthOfLIS(vector<int>& nums) {
int n = nums.size();
if (n < ) return n;
vector<int> dp(n, );
int res = INT_MIN;
for (int i = ; i < n; i++)
for (int j = ; j < i; j++)
if (nums[j] < nums[i])
dp[i] = max(dp[i], dp[j] + );
for (int i = ; i < n; i++)
res = max(res, dp[i]);
return res;
}
};

下面介绍O(NlogN)的算法。

该算法的主要思想可以用一个例子来说明。如果有两个数nums[a],nums[b],且nums[a] < nums[b],而dp[a] = dp[b],即以这两个数截止的最长递增子序列的长度是相等的,那么如果后面还有数,很明显我们要优先选择nums[a],因为它更小,选它则之后获得更长增长子序列的机会更大。

因此这里我们用len来记录目前已经发现的最长增长子序列的长度。

用一个int数组MinLastNumOfLenOf[k]来表示目前已经发现的,长度为k的最长增长子序列的最后一个元素的最小值。这里数组名字起这么长纯粹是为了好理解。

那么会发现MinLastNumOfLenOf中的值是递增的,即对于i < j有M[i] < M[j]。否则,假设M[j] < M[i] 且j > i,那么我们按照定义完全可以让M[i] 等于M[j]的值,这样会更优。同时,M[i]的值在算法运行过程中只降不增。

在实际进行中,len为当前我们已经发现的最长增长子序列的长度,而MinLastNumOfLenOf[len]则记录了该最长子序列的最后一个元素(最小的)。

当我们进行到nums[i]时,如果nums[i] > MinLastNumOfLenOf[len],则明显我们获得了一个更长的增长子序列,因此将len加一,然后令MinLastNumOfLenOf[len] = nums[i]。

否则,如果nums[i] < MinLastNumOfLenOf[len],则我们可以找到一个最小的下标k,k满足MinLastNumOfLenOf[k] > nums[i],然后我们将MinLastNumOfLenOf[k]更新为nums[i]的值。这里,因为M数组的值是递增的,我们可以用二分查找来找到这个下标,复杂度为O(logn)。

可能你会有疑问,我们都已经发现了长度为len的子序列了,为什么还要更新前面MinLastNumOfLenOf[k]的值呢?因为实际上M[1]...M[len]串连起来就是我们的最长增长子序列。但是,如果我们将后续的一个数nums[i] < M[len]更新到M数组中,假设更新到了M[k],则明显,M[1]...M[k]仍然是一个最长增长子序列,而M[1]...到M[k+1]以及后面的元素就不是了,因为我们新插入的这个数是后加进来的,也就是说在nums[i]中的下标i是要大于M[k+1]...M[len]这些数在nums中的下标的。每次我们更新M数组时总会有这个规律,而且,当我们将M[k]更新得更小后,则M[k+1]被更新的机会就更大,毕竟M[k]不可能无限小下去。而且,虽然M[k]更新后,M[1]...M[len]就无法构成一个合法的增长子序列,但我们已经用len保存了当前所发现的最长值,因此并无大碍。以上过程一直持续到我们找到一个新的数,而它仅小于M[len]时,M[len]被更新了,此时我们终于找到了一个长度为len且最后一个元素更小的子序列,而这使得后续找到更长的子序列的机会更大了!

 class Solution {
public:
int findInd(vector<int>& d, int num)
{
int left = , right = d.size() - ;
while (left < right)
{
int mid = (left + right) >> ;
if (d[mid] < num) left = mid + ;
else if (mid != && d[mid - ] < num)
return mid;
else right = mid;
}
return left;
}
int lengthOfLIS(vector<int>& nums) {
int n = nums.size();
if (n < ) return n;
int len = ;
vector<int> MinLastNumOfLenOf(, );
MinLastNumOfLenOf[len] = nums[];
for (int i = ; i < n; i++)
{
if (nums[i] > MinLastNumOfLenOf[len])
{
MinLastNumOfLenOf.push_back(nums[i]);
len++;
}
else if (nums[i] < MinLastNumOfLenOf[len])
MinLastNumOfLenOf[findInd(MinLastNumOfLenOf, nums[i])] = nums[i];
}
return len;
}
};

Longest Increasing Subsequence - LeetCode的更多相关文章

  1. [LeetCode] Longest Increasing Subsequence 最长递增子序列

    Given an unsorted array of integers, find the length of longest increasing subsequence. For example, ...

  2. leetcode@ [300] Longest Increasing Subsequence (记忆化搜索)

    https://leetcode.com/problems/longest-increasing-subsequence/ Given an unsorted array of integers, f ...

  3. [LeetCode] Number of Longest Increasing Subsequence 最长递增序列的个数

    Given an unsorted array of integers, find the number of longest increasing subsequence. Example 1: I ...

  4. LeetCode Number of Longest Increasing Subsequence

    原题链接在这里:https://leetcode.com/problems/number-of-longest-increasing-subsequence/description/ 题目: Give ...

  5. [LeetCode] 300. Longest Increasing Subsequence 最长递增子序列

    Given an unsorted array of integers, find the length of longest increasing subsequence. Example: Inp ...

  6. [LeetCode] 673. Number of Longest Increasing Subsequence 最长递增序列的个数

    Given an unsorted array of integers, find the number of longest increasing subsequence. Example 1: I ...

  7. 【LeetCode】673. Number of Longest Increasing Subsequence 解题报告(Python)

    [LeetCode]673. Number of Longest Increasing Subsequence 解题报告(Python) 标签(空格分隔): LeetCode 题目地址:https:/ ...

  8. 【LeetCode】300. Longest Increasing Subsequence 解题报告(Python & C++)

    作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxuemingzhu.cn/ 目录 题目描述 题目大意 解题方法 日期 题目地址:https://leetcode.c ...

  9. Leetcode 300 Longest Increasing Subsequence

    Given an unsorted array of integers, find the length of longest increasing subsequence. For example, ...

随机推荐

  1. java并发面试题-基础

    多线程 java中有几种方法可以实现一个线程? 1.直接继承thread类:2.实现runnable接口: 如何停止一个正在运行的线程?可以使用正在运行的线程,支持线程中断,通常是定义一个volati ...

  2. Intellij idea 出现错误 error:java: 无效的源发行版: 8解决方法

    这是由于jdk的版本与项目的要求不一致造成的,如果是maven项目,首先查看一下pom.xml,以我的项目为例: <build> <plugins> <plugin> ...

  3. 【Luogu P1661】扩散

    题目: 一个点每过一个单位时间就会向四个方向扩散一个距离,如图. 两个点$a$.$b$连通,记作$e(a,b)$,当且仅当$a$.$b$的扩散区域有公共部分.连通块的定义是块内的任意两个点$u$.$v ...

  4. 系统中同时安装sql2005 和 sql2008 R2 提示要删除SQL Server 2005 Express

    修改注册表:HKLM\Software\Microsoft\Microsoft SQL Server\90\Tools\ShellSEM,把 ShellSEM重命名即可 如果是64位机器 在  HKL ...

  5. docker常用命令整理

    Docker常见用法整理 Docker 使用客户端-服务器 (C/S) 架构模式,使用远程API来管理和创建Docker容器. Docker 容器通过 Docker 镜像来创建.容器与镜像的关系类似于 ...

  6. selenium webdriver——元素操作

    #Author:xiaoxiao from selenium import webdriver import time def abcd(): driver = webdriver.Firefox() ...

  7. 201621123034 《Java程序设计》第3周学习总结

    1. 本周学习总结 初学面向对象,会学习到很多碎片化的概念与知识.尝试学会使用思维导图将这些碎片化的概念.知识点组织起来.请使用工具画出本周学习到的知识点及知识点之间的联系.步骤如下: 1.1 写出你 ...

  8. monolog使用

    安装composer curl -sS https://getcomposer.org/installer | phpmv composer.phar /usr/local/bin/composer ...

  9. [UOJ#132][BZOJ4200][luogu_P2304][NOI2015]小园丁与老司机

    [UOJ#132][BZOJ4200][luogu_P2304][NOI2015]小园丁与老司机 试题描述 小园丁 Mr. S 负责看管一片田野,田野可以看作一个二维平面.田野上有 \(n\) 棵许愿 ...

  10. rmmod: can't change directory to “3.4.79+”,no such file or directory

    直接在/lib/modules目录下,在该目录下建立3.4.79+这个文件夹即可