算法新手,刷力扣遇到这题,搞了半天终于搞懂了,来这记录一下,欢迎大家交流指点。

题目描述:

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

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

解法一:暴力递归

  不解释,先暴力搞一下。(时间复杂度O(n^3),不行)

 1 class Solution {
2 public:
3 int l(vector<int>& nums) { // 返回以nums[0]开头的最长递增序列长度
4 if (nums.size() < 2)
5 return nums.size();
6 int max_len = 1;
7 for (int i = 1; i < nums.size(); ++ i)
8 if (nums[i] > nums[0]) {
9 vector<int> t{nums.begin() + i, nums.end()};
10 max_len = max(max_len, l(t) + 1);
11 }
12 return max_len;
13 }
14 int lengthOfLIS(vector<int>& nums) { // 所有序列遍历一遍
15 int max_len = 1;
16 for (i = 0; i < nums.size(); ++ i) {
17 vector<int> t(nums.begin() + i, nums.end());
18 max_len = max(max_len, l(t));
19 }
20 return max_len;
21 }
22 };

  小优化一下,记忆化搜索。(还是不行,时间复杂度还是太高)

 1 class Solution {
2 public:
3 int l(unordered_map<int, int>& map, vector<int>& nums) {
4 if (nums.size() < 2)
5 return nums.size();
6 if (map.find(nums[0]) != map.end()) // 如果已经知道了以某个数开头的元素的最长序列数,直接返回
7 return map[nums[0]];
8 int max_len = 1;
9 for (int i = 1; i < nums.size(); ++ i)
10 if (nums[i] > nums[0]) {
11 vector<int> t{nums.begin() + i, nums.end()};
12 max_len = max(max_len, l(map, t) + 1);
13 }
14 map[nums[0]] = max_len; // 记录以某个数开头的最长递增序列长度
15 return max_len;
16 }
17 int lengthOfLIS(vector<int>& nums) {
18 int max_len = 1;
19 unordered_map<int, int> map; // 哈希表,<开头的数,最长递归序列长度>
20 for (int i = 0; i < nums.size(); ++ i) {
21 vector<int> t(nums.begin() + i, nums.end());
22 max_len = max(max_len, l(map, t));
23 }
24 return max_len;
25 }
26 };

解法二:动态规划

  看来暴力是不行滴,还得动态规划。(时间复杂度O(n^2),AC了)

 1 class Solution {
2 public:
3 int lengthOfLIS(vector<int>& nums) {
4 vector<int> dp(nums.size(), 0); // 记录以nums[i]为结尾的最长递增子序列长度
5 for (int i = 1; i < nums.size(); ++ i)
6 for (int j = 0; j < i; ++ j) // 找一个最长的递增序列,接到它后面
7 if (nums[j] < nums[i])
8 dp[i] = max(dp[i], dp[j] + 1);
9 return *max_element(dp.begin(), dp.end()) + 1;
10 }
11 };

解法三:动态规划 + 二分查找

  动态规划方法是可行的,但是O(n^2)的时间复杂度还是较高,使用二分查找方法可以进一步优化。(时间复杂度O(nlogn),大提升)

 1 class Solution {
2 public:
3 int lengthOfLIS(vector<int>& nums) {
4 vector<int> dp(1, *nums.begin()); // 维护一个数组,用来存放最长的递增子序列
5 int left = 0, right = 0, mid = 0;
6 for (int i = 1; i < nums.size(); ++ i) { // 遍历一遍nums寻找每个元素在最长子序列中的插入位置
7 if (nums[i] > *(dp.end() - 1)) { // 如果当前元素比序列中所有元素都大,直接插到末尾
8 dp.push_back(nums[i]);
9 continue;
10 }
11 left = -1;              // 否则的话,替换掉序列中第一个大于等于它的元素,这样可以保证得到最长的递增序列
12 right = dp.size();
13 while (left + 1 != right) {
14 mid = (left + right) / 2;
15 if (dp[mid] >= nums[i])
16 right = mid;
17 else
18 left = mid;
19 }
20 dp[right] = nums[i];
21 }
22 return dp.size();
23 }
24 };

【LeetCode】300.最长递增子序列——暴力递归(O(n^3)),动态规划(O(n^2)),动态规划+二分法(O(nlogn))的更多相关文章

  1. Leetcode 673.最长递增子序列的个数

    最长递增子序列的个数 给定一个未排序的整数数组,找到最长递增子序列的个数. 示例 1: 输入: [1,3,5,4,7] 输出: 2 解释: 有两个最长递增子序列,分别是 [1, 3, 4, 7] 和[ ...

  2. Leetcode 300.最长上升子序列

    最长上升子序列 给定一个无序的整数数组,找到其中最长上升子序列的长度. 示例: 输入: [10,9,2,5,3,7,101,18] 输出: 4 解释: 最长的上升子序列是 [2,3,7,101],它的 ...

  3. [LeetCode] 300. 最长上升子序列 ☆☆☆(动态规划 二分)

    https://leetcode-cn.com/problems/longest-increasing-subsequence/solution/dong-tai-gui-hua-she-ji-fan ...

  4. Java实现 LeetCode 673 最长递增子序列的个数(递推)

    673. 最长递增子序列的个数 给定一个未排序的整数数组,找到最长递增子序列的个数. 示例 1: 输入: [1,3,5,4,7] 输出: 2 解释: 有两个最长递增子序列,分别是 [1, 3, 4, ...

  5. Java实现 LeetCode 300 最长上升子序列

    300. 最长上升子序列 给定一个无序的整数数组,找到其中最长上升子序列的长度. 示例: 输入: [10,9,2,5,3,7,101,18] 输出: 4 解释: 最长的上升子序列是 [2,3,7,10 ...

  6. leetcode 300最长上升子序列

    用递归DFS遍历所有组合肯定积分会超时,原因是有很多重复的操作,可以想象每次回溯后肯定会有重复操作.所以改用动态规划.建立一个vector<int>memo,初始化为1,memo[i]表示 ...

  7. Leetcode——300. 最长上升子序列

    题目描述:题目链接 给定一个无序的整数数组,找到其中最长上升子序列的长度. 示例: 输入: [10,9,2,5,3,7,101,18] 输出: 4 解释: 最长的上升子序列是 [2,3,7,101], ...

  8. LeetCode 300. 最长上升子序列(Longest Increasing Subsequence)

    题目描述 给出一个无序的整形数组,找到最长上升子序列的长度. 例如, 给出 [10, 9, 2, 5, 3, 7, 101, 18], 最长的上升子序列是 [2, 3, 7, 101],因此它的长度是 ...

  9. LeetCode 300——最长上升子序列

    1. 题目 2. 解答 2.1. 动态规划 我们定义状态 state[i] 表示以 nums[i] 为结尾元素的最长上升子序列的长度,那么状态转移方程为: \[state[i] = max(state ...

随机推荐

  1. HttpClient4.3教程 第四章 HTTP认证

    HttpClient既支持HTTP标准规范定义的认证模式,又支持一些广泛使用的非标准认证模式,比如NTLM和SPNEGO. 4.1.用户凭证 任何用户认证的过程,都需要一系列的凭证来确定用户的身份.最 ...

  2. C语言之----面向对象的方法实现链表的操作

    1 /* 2 * 详细运行过程: 本程序实现的是对链表的简单的操作,即链表的增 删 改 查 销毁 初始化 3 * 运用面向对象的思想,实现一个类op,op中包括了所有的链表操作方法 4 * 其他的程序 ...

  3. linux 常用命令(二)——(centos6.8-centos7)防火墙的启动、关闭

    centos 6.8 [centos6.5]: 查看chkconfig列表里面是否有iptables的服务: chkconfig | grep iptables 查看防火墙状态: service ip ...

  4. Go进阶--httptest

    目录 基本使用 扩展使用 接口context使用 模拟调用 测试覆盖率 参考 单元测试的原则,就是你所测试的函数方法,不要受到所依赖环境的影响,比如网络访问等,因为有时候我们运行单元测试的时候,并没有 ...

  5. Sentinel限流、降级配置详解

    安装Sentinel 下载sentinel-dashboard-1.8.2.jar 安装有jdk环境,8080端口未被占用 在jar包所在目录打开cmd,输入命令启动:java -jar sentin ...

  6. Java变量命名规范

    java命名规范 所有方法.变量.类名:见名知意 类成员变量:首字母小写.驼峰原则: 例如:lastName 第一个单词首字母小写,其余首字母大写 局部变量:首字母小写.驼峰原则 类名: 首字母小写. ...

  7. 作用域 作用域链 闭包 思想 JS/C++比较

    首先,我说的比较是指JS中这种思想/实现方式与C++编译原理中思想/实现方式的比较 参考链接:(比较易懂的介绍,我主要写个人理解) 作用域链: http://www.cnblogs.com/dolph ...

  8. Python - 进度条库 tqdm

    前言 在写生成器的时候,网上看到一个进度条库,感觉蛮有意思,记录下 这个库感觉只有在调试的时候会用到,不做深入学习 内置库,不需要安装 示例代码 from tqdm import tqdm for i ...

  9. jq的选择器中带有特殊符号无法获取元素

    因项目需要,将元素id命名为数组(array[i].string) 使用jq去获取该id的元素时,返回的是个undefined.即jq获取不到该元素,因为该元素中的id含有特殊字符"[&qu ...

  10. Spring整合MyBatis小结

    MyBatis在Spring中的配置 我们在Spring中写项目需要运用到数据库时,现在一般用的是MyBatis的框架来帮助我们书写代码,但是学习了SSM就要知道M指的就是MyBatis,在此,在Sp ...