力扣● 300.最长递增子序列 ● 674. 最长连续递增序列 ● 718. 最长重复子数组

2024-03-11 04:44

本文主要是介绍力扣● 300.最长递增子序列 ● 674. 最长连续递增序列 ● 718. 最长重复子数组,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

● 300.最长递增子序列

1.dp数组含义。dp[i]:以nums[i]为结尾的递增子序列的最大长度。

注意一定是以nums[i]结尾,如果dp[i]定义错误的话,暴力也不知道咋整。

2.递推公式。这道题与背包的单词划分比较像,一个单层循环是弄不出来的,外层循环i,内层循环j。即当前下标i的递增子序列长度,其实和i之前的下表j的子序列长度有关系。准确的说,i的递增子序列长度就是由[0,i-1]范围内的子序列长度得到的,要取他们中的一个最大值。分析如下:

如果nums[i]比nums[j]小或者相等,那么就不能把nums[i]拼到nums[j]结尾的序列上,即不能根据dp[j]得到dp[i],因为dp[i]一定是nums[i]为结尾的递增子序列的最大长度,nums[i]要比这个序列中的其他元素都大。

所以递推公式只考虑nums[i]比nums[j]大的情况,一开始,j=i-1,这个序列里面只有i自己,所以dp[i]=1,然后从后往前依次比较要不要加入j的子序列,有可能i加入j的这个子序列后长度更大,长度是dp[j]+1;也有可能加入后长度没变大,所以保持原样。

所以dp[i]应该取这两个的较大值:dp[i]=max(dp[j]+1,dp[i]);

代码随想录:注意这里不是要dp[i] 与 dp[j] + 1进行比较,而是我们要取dp[j] + 1的最大值(从后往前依次比较取得的)。

公式结合例子更好理解,举例:

nums = [10,9,2,5,3,7,101,18]

到元素3的时候,dp[4]一开始是1,意味着这个子序列只有自己;与5比较比5小,不能把自己拼到5的子序列后面,即不能由dp[3]得到,跳过;与2比较,可以拼到2后面,取max为2;比9大;比10大。所以dp[4]=2。

3.初始化。dp[i]的最大长度,一开始只有它自己一个元素组成一个序列,所以均初始化为1。

4.遍历顺序。根据递推公式过程,i从左到右,j从右到左。

5.打印。

代码如下。仍然注意dp数组的含义,最后不是返回dp[nums.size()-1],而是返回dp中的最大值。

class Solution {
public:int lengthOfLIS(vector<int>& nums) {vector<int> dp(nums.size(),1);int maxlen=1;for(int i=0;i<nums.size();++i){for(int j=i-1;j>=0;--j){if(nums[j]<nums[i]){dp[i]=max(dp[j]+1,dp[i]);maxlen=max(dp[i],maxlen);}}}return maxlen;}
};


● 674. 最长连续递增序列

这道题和上一道题的区别是:递增序列是连续的,那么不需要内层循环遍历[i-1,0],因为nums[i]只能拼接到nums[i-1]上面,即根据dp[i-1]得到dp[i],再前面的dp元素跟i扯不上关系。

所以一层循环就能实现,也不用比较max值。

class Solution {
public:int findLengthOfLCIS(vector<int>& nums) {vector<int> dp(nums.size(),1);int maxlen=1;for(int i=1;i<nums.size();++i){if(nums[i]>nums[i-1]){dp[i]=dp[i-1]+1;maxlen=max(maxlen,dp[i]);}}return maxlen;}
};


● 718. 最长重复子数组

这篇关于力扣● 300.最长递增子序列 ● 674. 最长连续递增序列 ● 718. 最长重复子数组的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Java 字符数组转字符串的常用方法

《Java字符数组转字符串的常用方法》文章总结了在Java中将字符数组转换为字符串的几种常用方法,包括使用String构造函数、String.valueOf()方法、StringBuilder以及A... 目录1. 使用String构造函数1.1 基本转换方法1.2 注意事项2. 使用String.valu

JAVA中整型数组、字符串数组、整型数和字符串 的创建与转换的方法

《JAVA中整型数组、字符串数组、整型数和字符串的创建与转换的方法》本文介绍了Java中字符串、字符数组和整型数组的创建方法,以及它们之间的转换方法,还详细讲解了字符串中的一些常用方法,如index... 目录一、字符串、字符数组和整型数组的创建1、字符串的创建方法1.1 通过引用字符数组来创建字符串1.2

vue如何监听对象或者数组某个属性的变化详解

《vue如何监听对象或者数组某个属性的变化详解》这篇文章主要给大家介绍了关于vue如何监听对象或者数组某个属性的变化,在Vue.js中可以通过watch监听属性变化并动态修改其他属性的值,watch通... 目录前言用watch监听深度监听使用计算属性watch和计算属性的区别在vue 3中使用watchE

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

poj2406(连续重复子串)

题意:判断串s是不是str^n,求str的最大长度。 解题思路:kmp可解,后缀数组的倍增算法超时。next[i]表示在第i位匹配失败后,自动跳转到next[i],所以1到next[n]这个串 等于 n-next[n]+1到n这个串。 代码如下; #include<iostream>#include<algorithm>#include<stdio.h>#include<math.

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

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

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

hdu 1166 敌兵布阵(树状数组 or 线段树)

题意是求一个线段的和,在线段上可以进行加减的修改。 树状数组的模板题。 代码: #include <stdio.h>#include <string.h>const int maxn = 50000 + 1;int c[maxn];int n;int lowbit(int x){return x & -x;}void add(int x, int num){while

uva 10131 最长子序列

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

XTU 1233 n个硬币连续m个正面个数(dp)

题面: Coins Problem Description: Duoxida buys a bottle of MaiDong from a vending machine and the machine give her n coins back. She places them in a line randomly showing head face or tail face o