C++二分查找算法的应用:最长递增子序列

涉及知识点

二分查找 单调映射

源码下载

点击下载源码

题目

给你一个整数数组 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

暴力解法

分析

m_vRet[i],记录以nums[i]结尾的最长严格递增系列。计算m_vRet[i]的方法:nums[j] < nums[i]中最大m_vRet[j]+1。如果不存在合法的j,则m_vRet[i]等于1。

下表以{1,2,6,5,4,5}

|---|---------------------------------------|-------------|
| i | m_vRet[0,i),加粗表示nums[j]<nums[i] | m_vRet[i] |
| 0 | | 1 |
| 1 | 1 | 2{1,2} |
| 2 | 1 2 | 3{1,2,6} |
| 3 | 1 23 | 3{1,2,5} |
| 4 | 1 23 3 | 3{1,2,4} |
| 5 | 1 2 3 3 3 | 4{1,2,4,5} |

两层循环,分别枚举i和j,故总时间复杂度是O(n2)。

代码

class Solution {

public:

int lengthOfLIS(vector<int>& nums) {

m_c = nums.size();

m_vRet.resize(m_c);

for (int i = 0; i < m_c; i++)

{

int iPreLen = 0;

for (int j = 0; j < i; j++)

{

if (nums[j] >= nums[i])

{

continue;

}

iPreLen = max(iPreLen, m_vRet[j]);

}

m_vRet[i] = iPreLen + 1;

}

return *std::max_element(m_vRet.begin(), m_vRet.end());

}

vector<int> m_vRet;

int m_c;

};

有序映射

分析

如果nums[j1] > nums[j2],且m_vRet[j1] <= m_vRet[j2],那么j1被j2淘汰。以nums[j]为key,m_vRet[j]为值,创建有序映射。key和value都是升序。

下表以{1,2,6,5,4,5,5}为列来说明,加粗表示新增加,删除线表示删除。

|------------------------------|
| {1,1} |
| {1,1},{2,2} |
| {1,1},{2,2},{6,3} |
| {1,1},{2,2},{5,3} ,{6,3} |
| {1,1},{2,2},{4,3} ,{5,3} |
| {1,1},{2,2},{4,3},{5,4} |
| {1,1},{2,2},{4,3},{5,4} |

代码

class Solution {

public:

int lengthOfLIS(vector<int>& nums) {

m_c = nums.size();

m_vRet.resize(m_c);

std::map<int, int> mValueLen = { {-1000 * 1000,0} };

for (int i = 0; i < m_c; i++)

{

//计算以nums[i]结尾的最长长度

auto it = mValueLen.lower_bound(nums[i]);

const int iLen = std::prev(it)->second + 1;

m_vRet[i] = iLen;

//如果nums[j] >= nums[j],且长度小于等于删除

auto ij = it;

for (; (ij != mValueLen.end()) && (ij->second <= iLen); ++ij);

mValueLen.erase(it, ij);

//如果nums[i]存在,说明旧值比当前值大,不能处理

if (!mValueLen.count(nums[i]))

{

mValueLen[nums[i]] = iLen;

}

}

return *std::max_element(m_vRet.begin(), m_vRet.end());

}

vector<int> m_vRet;

int m_c;

};

其它

学院课程

|---------------------------------------------------------------------------------------------------------------------------------|
| 基础算法的C++实现课程,请点击下面的CSDN学院的链接。讲义有算法详解。 |
| 2024年1月15之前完全免费,之后绝大部分免费 |
| https://edu.csdn.net/course/detail/38771 |
| C#入职培训 |
| 此课程的目的:让新同事更快完成从学生到C#程序员的转换,更快上手完成C#的开发工作。 |
| https://edu.csdn.net/course/detail/38768 |
| C++入职培训 |
| 让新同事更快完成从学生到C++程序员的转换,更快上手完成C++的开发工作。 |
| https://edu.csdn.net/course/detail/32049 |

运行验证环境

Win10 VS2022 Ck++17 或win7 VS2019 C++17

每天都补充正能量

|---------------|
| 好好学习,天天向上。 |
| 事无终始,无务多业。 |
| 是故置本不安者,无务丰末。 |

相关下载

如果你时间宝贵,只想看精华,请到CSDN下载频道下载《闻缺陷则喜算法册》doc版

https://download.csdn.net/download/he_zhidan/88348653

相关推荐
计算机小白一个5 分钟前
蓝桥杯 Java B 组之设计 LRU 缓存
java·算法·蓝桥杯
万事可爱^39 分钟前
HDBSCAN:密度自适应的层次聚类算法解析与实践
算法·机器学习·数据挖掘·聚类·hdbscan
黑不溜秋的40 分钟前
C++ 设计模式 - 策略模式
c++·设计模式·策略模式
大数据追光猿3 小时前
Python应用算法之贪心算法理解和实践
大数据·开发语言·人工智能·python·深度学习·算法·贪心算法
Dream it possible!3 小时前
LeetCode 热题 100_在排序数组中查找元素的第一个和最后一个位置(65_34_中等_C++)(二分查找)(一次二分查找+挨个搜索;两次二分查找)
c++·算法·leetcode
夏末秋也凉3 小时前
力扣-回溯-46 全排列
数据结构·算法·leetcode
南宫生3 小时前
力扣每日一题【算法学习day.132】
java·学习·算法·leetcode
柠石榴3 小时前
【练习】【回溯No.1】力扣 77. 组合
c++·算法·leetcode·回溯
Leuanghing3 小时前
【Leetcode】11. 盛最多水的容器
python·算法·leetcode
qy发大财3 小时前
加油站(力扣134)
算法·leetcode·职场和发展