前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >《剑指offer》第28天:最长上升子序列(高频)

《剑指offer》第28天:最长上升子序列(高频)

作者头像
程序员小浩
发布2020-09-14 17:06:00
4390
发布2020-09-14 17:06:00
举报
文章被收录于专栏:小浩算法小浩算法

01、题目分析

第300题:最长上升子序列

给定一个无序的整数数组,找到其中最长上升子序列的长度。

示例:

代码语言:javascript
复制
输入: [10,9,2,5,3,7,101,18]
输出: 4 
解释: 最长的上升子序列是 [2,3,7,101],它的长度是 4。

说明:

可能会有多种最长上升子序列的组合,你只需要输出对应的长度即可

02、题目图解

首先我们分析题目,要找的是最长上升子序列(Longest Increasing Subsequence,LIS)。因为题目中没有要求连续,所以 LIS可能是连续的,也可能是非连续的。 同时,LIS符合可以从其子问题的最优解来进行构建的条件。所以我们可以尝试用动态规划来进行求解。首先我们定义状态:

dp[i] :表示以nums[i]结尾的最长上升子序列的长度

我们假定nums为[1,9,5,9,3],如下图:

我们分两种情况进行讨论:

  • 如果nums[i]比前面的所有元素都小,那么dp[i]等于1(即它本身)(该结论正确)
  • 如果nums[i]前面存在比他小的元素nums[j],那么dp[i]就等于dp[j]+1(该结论错误,比如nums[3]>nums[0],即9>1,但是dp[3]并不等于dp[0]+1)

我们先初步得出上面的结论,但是我们发现了一些问题。**因为dp[i]前面比他小的元素,不一定只有一个!**

可能除了 nums[j],还包括 nums[k],nums[p] 等等等等。所以 dp[i] 除了可能等于 dp[j]+1,还有可能等于 dp[k]+1,dp[p]+1 等等等等。所以我们求 dp[i],需要找到 dp[j]+1,dp[k]+1,dp[p]+1 等等等等 中的最大值。(我在3个等等等等上都进行了加粗,主要是因为初学者非常容易在这里摔跟斗!这里强调的目的是希望能记住这道题型!) 即:

dp[i] = max(dp[j]+1,dp[k]+1,dp[p]+1,.....) 只要满足: nums[i] > nums[j] nums[i] > nums[k] nums[i] > nums[p] ....

最后,我们只需要找到dp数组中的最大值,就是我们要找的答案。

分析完毕,我们绘制成图:

03、Go语言示例

根据以上分析,可以得到代码如下:

代码语言:javascript
复制
func lengthOfLIS(nums []int) int {
 if len(nums) < 1 {
  return 0
 }
 dp := make([]int, len(nums))
 result := 1
 for i := 0; i < len(nums); i++ {
  dp[i] = 1
  for j := 0; j < i; j++ {
   if nums[j] < nums[i] {
    dp[i] = max(dp[j]+1, dp[i])
   }
  }
  result = max(result, dp[i])
 }
 return result
}

func max(a, b int) int {
 if a > b {
  return a
 }
 return b
}
本文参与 腾讯云自媒体分享计划,分享自微信公众号。
原始发表:2020-09-11,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 小浩算法 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 02、题目图解
  • 03、Go语言示例
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档