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

相关文章

第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

软考系统规划与管理师考试证书含金量高吗?

2024年软考系统规划与管理师考试报名时间节点: 报名时间:2024年上半年软考将于3月中旬陆续开始报名 考试时间:上半年5月25日到28日,下半年11月9日到12日 分数线:所有科目成绩均须达到45分以上(包括45分)方可通过考试 成绩查询:可在“中国计算机技术职业资格网”上查询软考成绩 出成绩时间:预计在11月左右 证书领取时间:一般在考试成绩公布后3~4个月,各地领取时间有所不同

uva 10131 最长子序列

题意: 给大象的体重和智商,求体重按从大到小,智商从高到低的最长子序列,并输出路径。 代码: #include <iostream>#include <cstdio>#include <cstdlib>#include <algorithm>#include <cstring>#include <cmath>#include <stack>#include <vect

poj 3190 优先队列+贪心

题意: 有n头牛,分别给他们挤奶的时间。 然后每头牛挤奶的时候都要在一个stall里面,并且每个stall每次只能占用一头牛。 问最少需要多少个stall,并输出每头牛所在的stall。 e.g 样例: INPUT: 51 102 43 65 84 7 OUTPUT: 412324 HINT: Explanation of the s