300. 最长递增子序列 动态规划O(n2) 贪心+二分 O(nlogn)

2023-11-28 13:38

本文主要是介绍300. 最长递增子序列 动态规划O(n2) 贪心+二分 O(nlogn),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

思路

  1. 动态规划,定义状态 d p [ i ] dp[i] dp[i]代表以 n u m s [ i ] nums[i] nums[i]结尾的最长递增子序列的长度,这样的话 d p [ i ] dp[i] dp[i]和其子问题 d p [ k ] , 0 < = k < i dp[k],0<=k<i dp[k],0<=k<i便产生了联系、
    • 状态转移方程 d p [ i ] = m a x ( d p [ k ] + 1 ) , 0 < = k < i i f ( n u m s [ i ] > n u m s [ k ] ) dp[i]=max(dp[k]+1),0<=k<i if(nums[i]>nums[k]) dp[i]=max(dp[k]+1),0<=k<iif(nums[i]>nums[k])
    • 时间复杂度 O ( n 2 ) O(n^2) O(n2)
    • 空间复杂度 O ( n ) O(n) O(n)
  2. 官方给的 O ( n l o g n ) O(nlogn) O(nlogn)的解法,采用贪心和二分。
    • 维护一个 d [ ] d[] d[]数组, d [ i ] d[i] d[i]代表长度为 i i i的最长递增子序列的最后一个数字的值, l e n len len代表当前得到的最长序列的长度。这样做是基于贪心的思想,我们希望我们的序列的数字增长的尽可能的慢,这样的话我们后面就可以尽可能多的增加数字。
    • 如果当前的数值 n u m s [ i ] > d [ l e n ] nums[i]>d[len] nums[i]>d[len],我们更新 d [ + + l e n ] = n u m s [ i ] d[++len]=nums[i] d[++len]=nums[i],否则,我们需要更新 d d d数组前面的数值, d [ ] d[] d[]数组在这里是单调递增的,如果 d [ ] d[] d[]数组不是单调递增的,假设一个长度为 n n n的递增子序列和一个长度为 n + 1 n+1 n+1的递增子序列,且有 d [ n + 1 ] < d [ n ] d[n+1]<d[n] d[n+1]<d[n],并且我们知道 d [ n ] , d [ n + 1 ] d[n],d[n+1] d[n],d[n+1]分别代表着长度为 n n n n + 1 n+1 n+1的递增子序列,那么 d [ n + 1 ] d[n+1] d[n+1]的倒数第二个数字肯定在 d [ n ] d[n] d[n]这个数字要小, d [ n + 1 ] 和 d [ n ] d[n+1]和d[n] d[n+1]d[n]之间只有一个数字长度的差距, d [ n ] d[n] d[n]显然应该不为 d [ n ] d[n] d[n]而为 d [ n + 1 ] d[n+1] d[n+1]所代表序列的倒数第二个数字。官方题解的证明。我们可以用二分查找的方式找到我们的满足条件 d [ i − 1 ] < = n u m s [ j ] < n u m s [ i ] d[i-1]<=nums[j]<nums[i] d[i1]<=nums[j]<nums[i] i i i,我们去更新这个 d [ i ] d[i] d[i]维护好 d [ ] d[] d[]数组,这样最后的 d [ l e n ] d[len] d[len]就是我们的答案了。

时间复杂度 O ( n 2 ) O(n^2) O(n2),空间复杂度 O ( n ) O(n) O(n)

// 时间复杂度$O(n^2)$
class Solution {
public:vector<int>res;int lengthOfLIS(vector<int>& nums) {int n=nums.size();if(n==0){return 0;}res.resize(n,1);// res[0]=1;for(int i=1;i<n;i++){for(int j=0;j<i;j++){if(nums[i]>nums[j])res[i]=max(res[i],res[j]+1);}}int ans=res[0];for(int i=1;i<n;i++){ans=max(ans,res[i]);}return ans;}
};

// 时间复杂度O(nlogn)
class Solution {
public:vector<int>d;int len;int lengthOfLIS(vector<int>& nums) {int n=nums.size();if(n==0){return 0;}d.resize(n+1,0);len=1;                  // 当前的最长的单调递增子序列的长度d[0]=0,d[1]=nums[0];for(int i=1;i<n;i++){if(nums[i]>d[len]){d[++len]=nums[i];}else{int left=1,right=len;int mid;// 怎么找到数组中满足条件nums[i-1]<d[j]<=nums[i]的iint pos=0;while(left<=right){mid=(left+right)/2;if(d[mid]<nums[i]){pos=mid;left=mid+1;}else{right=mid-1;}}// 官方题解这里的写法不错d[pos+1]=nums[i];}}return len;}
};

这篇关于300. 最长递增子序列 动态规划O(n2) 贪心+二分 O(nlogn)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/429788

相关文章

如何用Python绘制简易动态圣诞树

《如何用Python绘制简易动态圣诞树》这篇文章主要给大家介绍了关于如何用Python绘制简易动态圣诞树,文中讲解了如何通过编写代码来实现特定的效果,包括代码的编写技巧和效果的展示,需要的朋友可以参考... 目录代码:效果:总结 代码:import randomimport timefrom math

Java中JSON字符串反序列化(动态泛型)

《Java中JSON字符串反序列化(动态泛型)》文章讨论了在定时任务中使用反射调用目标对象时处理动态参数的问题,通过将方法参数存储为JSON字符串并进行反序列化,可以实现动态调用,然而,这种方式容易导... 需求:定时任务扫描,反射调用目标对象,但是,方法的传参不是固定的。方案一:将方法参数存成jsON字

.NET利用C#字节流动态操作Excel文件

《.NET利用C#字节流动态操作Excel文件》在.NET开发中,通过字节流动态操作Excel文件提供了一种高效且灵活的方式处理数据,本文将演示如何在.NET平台使用C#通过字节流创建,读取,编辑及保... 目录用C#创建并保存Excel工作簿为字节流用C#通过字节流直接读取Excel文件数据用C#通过字节

第10章 中断和动态时钟显示

第10章 中断和动态时钟显示 从本章开始,按照书籍的划分,第10章开始就进入保护模式(Protected Mode)部分了,感觉从这里开始难度突然就增加了。 书中介绍了为什么有中断(Interrupt)的设计,中断的几种方式:外部硬件中断、内部中断和软中断。通过中断做了一个会走的时钟和屏幕上输入字符的程序。 我自己理解中断的一些作用: 为了更好的利用处理器的性能。协同快速和慢速设备一起工作

hdu2241(二分+合并数组)

题意:判断是否存在a+b+c = x,a,b,c分别属于集合A,B,C 如果用暴力会超时,所以这里用到了数组合并,将b,c数组合并成d,d数组存的是b,c数组元素的和,然后对d数组进行二分就可以了 代码如下(附注释): #include<iostream>#include<algorithm>#include<cstring>#include<stack>#include<que

hdu2289(简单二分)

虽说是简单二分,但是我还是wa死了  题意:已知圆台的体积,求高度 首先要知道圆台体积怎么求:设上下底的半径分别为r1,r2,高为h,V = PI*(r1*r1+r1*r2+r2*r2)*h/3 然后以h进行二分 代码如下: #include<iostream>#include<algorithm>#include<cstring>#include<stack>#includ

动态规划---打家劫舍

题目: 你是一个专业的小偷,计划偷窃沿街的房屋。每间房内都藏有一定的现金,影响你偷窃的唯一制约因素就是相邻的房屋装有相互连通的防盗系统,如果两间相邻的房屋在同一晚上被小偷闯入,系统会自动报警。 给定一个代表每个房屋存放金额的非负整数数组,计算你 不触动警报装置的情况下 ,一夜之内能够偷窃到的最高金额。 思路: 动态规划五部曲: 1.确定dp数组及含义 dp数组是一维数组,dp[i]代表

poj3261(可重复k次的最长子串)

题意:可重复k次的最长子串 解题思路:求所有区间[x,x+k-1]中的最小值的最大值。求sa时间复杂度Nlog(N),求最值时间复杂度N*N,但实际复杂度很低。题目数据也比较水,不然估计过不了。 代码入下: #include<iostream>#include<algorithm>#include<stdio.h>#include<math.h>#include<cstring

usaco 1.3 Barn Repair(贪心)

思路:用上M块木板时有 M-1 个间隙。目标是让总间隙最大。将相邻两个有牛的牛棚之间间隔的牛棚数排序,选取最大的M-1个作为间隙,其余地方用木板盖住。 做法: 1.若,板(M) 的数目大于或等于 牛棚中有牛的数目(C),则 目测 给每个牛牛发一个板就为最小的需求~ 2.否则,先对 牛牛们的门牌号排序,然后 用一个数组 blank[ ] 记录两门牌号之间的距离,然后 用数组 an

poj 3974 and hdu 3068 最长回文串的O(n)解法(Manacher算法)

求一段字符串中的最长回文串。 因为数据量比较大,用原来的O(n^2)会爆。 小白上的O(n^2)解法代码:TLE啦~ #include<stdio.h>#include<string.h>const int Maxn = 1000000;char s[Maxn];int main(){char e[] = {"END"};while(scanf("%s", s) != EO