可视化图解算法75:最长上升子序列(最长递增子序列)

1.题目

描述

给定一个长度为 n 的数组 arr,求它的最长严格上升子序列的长度。

所谓子序列,指一个数组删掉一些数(也可以不删)之后,形成的新数组。例如 1,5,3,7,3 数组,其子序列有:1,3,37 等。但 1,11,3,5 则不是它的子序列。

我们定义一个序列是 严格上升 的,当且仅当该序列不存在 两个下标 i 和 j 满足 i<j 且 arr~i~ ≥arr~j~

数据范围: 0≤n≤1000

要求:时间复杂度 O(n^2^), 空间复杂度 O(n)

示例1

输入:

6,3,1,5,2,3,7

返回值:

4

说明:

该数组最长上升子序列为 1,2,3,7 ,长度为4

2. 题解思路

本题求解的是最长子序列,因此需要了解子序列与子串的区别。

子串和子序列是计算机科学中常见的两个概念,它们都与字符串或序列的部分内容有关,但存在明显的区别。

子串(Substring) 定义:子串是指原字符串中连续的一段字符组成的序列。换句话说,子串是从原字符串中选取的一个或多个相邻字符构成的新字符串。

特点:子串是连续的,即子串中的字符在原字符串中是连续排列的。子串的长度至少为1,也可以是整个原字符串(即原字符串本身也是它自己的子串)。空字符串(即不包含任何字符的字符串)在某些定义下也被视为任何字符串的子串,但这取决于具体的上下文和定义。 示例:在字符串"abcdefg"中,"abc"、"def"和"cde"都是合法的子串。

子序列(Subsequence 定义:子序列是从原序列中任意选取若干个元素组成的新序列,且这些元素在新序列中的相对顺序与原序列相同,但是允许跳过某些元素。

特点:子序列不一定连续,即子序列中的元素在原序列中可以是不连续的。子序列可以为空,即不选取任何元素。子序列也可以包含原序列的所有元素(此时子序列就等于原序列自身)。 示例:在字符串"abcdefg"中,"aceg"、"adg"和"bdf"都是合法的子序列。

具体思路如下:

如果文字描述的不太清楚,你可以参考视频的详细讲解。

3.编码实现

核心代码如下:

Go 复制代码
func LIS(arr []int) int {
	//特殊情况
	if len(arr) == 0 {
		return 0
	}
	//1.定义状态.	i:数组arr的下标; dp[i]:数组arr[0:i]区间内的最长递增子序列
	dp := make([]int, len(arr))
	//2.初始化边界条件:dp[i]=1,对于每一个元素来说,如果只有此元素本身,最长递增子序列为1
	for i := 0; i < len(dp); i++ {
		dp[i] = 1
	}
	res := 1
	//3.确定递推公式:dp[i]=max(dp[j]+1,dp[i]);arr[i]>arr[j]. 其中,arr[j]为当前元素arr[i]左侧的所有元素
	for i := 1; i < len(arr); i++ {
		//查找上升序列
		for j := 0; j < i; j++ {
			//数组序列上升则更新dp
			if arr[i] > arr[j] {
				dp[i] = max(dp[j]+1, dp[i]) //i需要与多个j进行比较,因此需获取其中最大的dp[i]
			}
		}
		//找到(更新)最大长度
		res = max(res, dp[i])
	}
	//4.输出结果:dp数组中最大的元素
	return res
}

func max(a int, b int) int {
	if a >= b {
		return a
	}
	return b
}

具体完整代码你可以参考下面视频的详细讲解。

4.总结

本题求解的是最长子序列,需要明白的是子序列是不连续的。通过动态规划来解决时需要清楚变量i、dpi数组的含义,理解递推公式的含义:dpi=max(dpj+1,dpi);arri>arrj. 其中,arrj为当前元素arri左侧的所有元素。

《数据结构与算法》深度精讲课程正式上线啦!7 大核心算法模块全解析:

  ✅ 链表

  ✅ 二叉树

  ✅ 二分查找、排序

  ✅ 堆、栈、队列

  ✅ 回溯算法

  ✅ 哈希算法

  ✅ 动态规划

无论你是备战笔试面试 、提升代码效率 ,还是突破技术瓶颈,这套课程都将为你构建扎实的算法思维底座。🔥立即加入学习打卡,与千名开发者共同进阶!

对于LeetCode数据结构与算法,我们总结了一套【可视化+图解】方法,依据此方法来解决相关问题,算法变得易于理解,写出来的代码可读性高也不容易出错。具体也可以参考视频详细讲解。

今日佳句:衣带渐宽终不悔,为伊消得人憔悴。

相关推荐
happymaker06268 小时前
LeetCodeHot100——42.接雨水
算法
阿正的梦工坊9 小时前
【Rust】07-错误处理:Option、Result 与 ? 运算符
开发语言·算法·rust
烬羽9 小时前
从零理解树与二叉树:用 JS 带你手撕遍历和递归
javascript·数据结构
YHL9 小时前
🚀从零理解树与二叉树 —— 概念、实现与遍历
前端·javascript·数据结构
JieE2129 小时前
JS 到底有多少种数据类型?从ECMA规范到内存本质,一文彻底搞懂
javascript·数据结构·面试
努力努力再努力wz10 小时前
【内存管理与高并发内存池系列】从 mmap 到 malloc:文件映射、匿名映射与 glibc 内存分配机制详解
linux·c语言·数据结构·数据库·c++·qt·链表
八解毒剂10 小时前
数据结构-平衡二叉树——对二叉搜索树的优化
数据结构·c++·算法
运行时记录10 小时前
别再手动写提示词了 — SkillOpt 让技能文档自己进化
算法
啦啦啦啦啦zzzz11 小时前
算法总结(二分查找、双指针)
c++·算法
qq_85730581911 小时前
python语法
开发语言·python·算法