题目链接 -> Leetcode -300.最长递增子序列
Leetcode -300.最长递增子序列
题目:给你一个整数数组 nums ,找到其中最长严格递增子序列的长度。
子序列 是由数组派生而来的序列,删除(或不删除)数组中的元素而不改变其余元素的顺序。
例如,[3, 6, 2, 7] 是数组[0, 3, 1, 6, 2, 2, 7] 的子序列。
示例 1:
输入:nums = [10, 9, 2, 5, 3, 7, 101, 18]
输出:4
解释:最长递增子序列是[2, 3, 7, 101],因此长度为 4 。
示例 2:
输入:nums = [0, 1, 0, 3, 2, 3]
输出:4
示例 3:
输入:nums = [7, 7, 7, 7, 7, 7, 7]
输出:1
提示:
思路:
因此,我们仅需找到满足要求的最大的 dp[j] + 1 即可;
综上, dp[i] = max(dp[j] + 1, dp[i]) ,其中 0 <= j <= i - 1 && nums[j] < nums[i];
代码如下:
class Solution { public: int lengthOfLIS(vector& nums) { // 动态规划 // dp[i] 表示:以 i 位置元素为结尾的「所有子序列」中,最长递增子序列的长度 vector dp(nums.size(), 1); int ret = 0; for (int i = nums.size() - 1; i >= 0; i--) { for (int j = i + 1; j < nums.size(); j++) { if (nums[j] > nums[i]) { dp[i] = max(dp[j] + 1, dp[i]); } } ret = max(dp[i], ret); } return ret; } };
题目链接 -> Leetcode -376.摆动序列
Leetcode -376.摆动序列
题目:如果连续数字之间的差严格地在正数和负数之间交替,则数字序列称为 摆动序列 。第一个差(如果存在的话)可能是正数或负数。仅有一个元素或者含两个不等元素的序列也视作摆动序列。
例如,[1, 7, 4, 9, 2, 5] 是一个 摆动序列 ,因为差值(6, -3, 5, -7, 3) 是正负交替出现的。
相反,[1, 4, 7, 2, 5] 和[1, 7, 4, 5, 5] 不是摆动序列,第一个序列是因为它的前两个差值都是正数,第二个序列是因为它的最后一个差值为零。
子序列 可以通过从原始序列中删除一些(也可以不删除)元素来获得,剩下的元素保持其原始顺序。
给你一个整数数组 nums ,返回 nums 中作为 摆动序列 的 最长子序列的长度 。
示例 1:
输入:nums = [1, 7, 4, 9, 2, 5]
输出:6
解释:整个序列均为摆动序列,各元素之间的差值为(6, -3, 5, -7, 3) 。
示例 2:
输入:nums = [1, 17, 5, 10, 13, 15, 10, 5, 16, 8]
输出:7
解释:这个序列包含几个长度为 7 摆动序列。
其中一个是[1, 17, 10, 13, 10, 16, 8] ,各元素之间的差值为(16, -7, 3, -3, 6, -8) 。
示例 3:
输入:nums = [1, 2, 3, 4, 5, 6, 7, 8, 9]
输出:2
提示:
思路:
但是,问题来了,如果状态表示这样定义的话,以 i 位置为结尾的最长摆动序列的长度我们没法从之前的状态推导出来。因为我们不知道前一个最长摆动序列的结尾处是递增的,还是递减的。因此,我们需要状态表示能表示多一点的信息:要能让我们知道这一个最长摆动序列的结尾是递增的还是递减的。
所以我们应该定义两个 dp 表:
i. 子序列长度为 1 :只能自己了,此时 f[i] = 1;这种情况不需要处理,我们在初始化的时候将 dp 表初始化为 1 即可 ;
ii. 子序列长度大于 1 :因为结尾要呈现上升趋势,因此需要 nums[j] < nums[i] 。在满足这个条件下, j 结尾需要呈现下降状态,最长的摆动序列就是 g[j] + 1 。因此我们要找出所有满足条件下的最大的 g[j] + 1 。
综上,因为题目需要返回的是最长子序列的长度所以 f[i] 的状态转移方程为 f[i] = max(g[j] + 1, f[i]) ,注意使用 g[j] 时需要判断;
i. 子序列长度为 1 :只能自己了,此时 g[i] = 1 ;这种情况不需要处理,我们在初始化的时候将 dp 表初始化为 1 即可 ;
ii. 子序列长度大于 1 :因为结尾要呈现下降趋势,因此需要 nums[j] > nums[i] 。在满足这个条件下, j 结尾需要呈现上升状态,因此最长的摆动序列就是 f[j] + 1 。因此我们要找出所有满足条件下的最大的 f[j] + 1 。
综上, g[i] = max(f[j] + 1, g[i]) ,注意使用 f[j] 时需要判断;
代码如下:
class Solution { public: int wiggleMaxLength(vector& nums) { int n = nums.size(); // f 表存最后呈现“上升”趋势的最长摆动子序列的长度 // g 表存最后呈现“下降”趋势的最长摆动子序列的长度 vector f(n, 1), g(n, 1); int ans = 1; for(int i = 1; i < n; i++) { for(int j = i - 1; j >= 0; j--) { // 如果是上升趋势,f[i] 就是 g 表中的最长摆动子序列的长度 + 1,因为 g 表是呈下降趋势的;下同 if(nums[i] > nums[j]) f[i] = max(g[j] + 1, f[i]); else if(nums[i] < nums[j]) g[i] = max(f[j] + 1, g[i]); } ans = max(g[i], max(f[i], ans)); } return ans; } };
题目链接 -> Leetcode -673.最长递增子序列的个数
Leetcode -673.最长递增子序列的个数
题目:给定一个未排序的整数数组 nums , 返回最长递增子序列的个数 。
注意 这个数列必须是 严格 递增的。
示例 1:
输入: [1, 3, 5, 4, 7]
输出 : 2
解释 : 有两个最长递增子序列,分别是[1, 3, 4, 7] 和[1, 3, 5, 7]。
示例 2 :
输入 : [2, 2, 2, 2, 2]
输出 : 5
解释 : 最长递增子序列的长度是1,并且存在5个子序列的长度为1,因此输出5。
提示 :
思路:
因此,我们解决这个问题需要两个状态,一个是「长度」,一个是「个数」:
综上所述,对于 len[i] ,我们可以得到状态转移方程为:
在知道每一个位置结尾的最长递增子序列的长度时,我们来看看能否得到 count[i] :
综上所述,对于 count[i] ,我们可以得到状态转移方程为:
代码如下:
class Solution { public: int findNumberOfLIS(vector& nums) { int n = nums.size(); // len[i] 表示以 i 位置结尾的最长递增子序列的长度 // count[i] 表示以 i 位置结尾的最长递增子序列的个数 vector len(n, 1), count(n, 1); int retlen = 1, retcount = 1; for (int i = 1; i < n; i++) { for (int j = i - 1; j >= 0; j--) { if (nums[i] > nums[j]) { // 贪心 if (len[j] + 1 > len[i]) { len[i] = len[j] + 1; count[i] = count[j]; } else if (len[j] + 1 == len[i]) { count[i] += count[j]; } } } // 贪心 if (retlen == len[i]) { retcount += count[i]; } else if (len[i] > retlen) { retlen = len[i]; retcount = count[i]; } } return retcount; } };
题目链接 -> Leetcode -646.最长数对链
Leetcode -646.最长数对链
题目:给你一个由 n 个数对组成的数对数组 pairs ,其中 pairs[i] = [lefti, righti] 且 lefti < righti 。
现在,我们定义一种 跟随 关系,当且仅当 b < c 时,数对 p2 = [c, d] 才可以跟在 p1 = [a, b] 后面。我们用这种形式来构造 数对链 。
找出并返回能够形成的 最长数对链的长度 。
你不需要用到所有的数对,你可以以任何顺序选择其中的一些数对来构造。
示例 1:
输入:pairs = [[1, 2], [2, 3], [3, 4]]
输出:2
解释:最长的数对链是[1, 2] ->[3, 4] 。
示例 2:
输入:pairs = [[1, 2], [7, 8], [4, 5]]
输出:3
解释:最长的数对链是[1, 2] ->[4, 5] ->[7, 8] 。
提示:
思路:本题思路与第一题 最长递增子序列 的思路类似,我们只需要将数对按照第一个元素排一下序即可处理;
代码如下:
class Solution { public: int findLongestChain(vector>& pairs) { sort(pairs.begin(), pairs.end()); // 对第一个元素排序 int n = pairs.size(); vector dp(n, 1); int ans = 1; // dp[i] 表⽰以 i 位置的数对为结尾时,最⻓数对链的⻓度 for(int i = 1; i < n; i++) { for(int j = i - 1; j >= 0; j--) { if(pairs[i][0] > pairs[j][1]) dp[i] = max(dp[j] + 1, dp[i]); } ans = max(dp[i], ans); } return ans; } };
题目链接 -> Leetcode -1218.最长定差子序列
Leetcode -1218.最长定差子序列
题目:给你一个整数数组 arr 和一个整数 difference,请你找出并返回 arr 中最长等差子序列的长度,该子序列中相邻元素之间的差等于 difference 。
子序列 是指在不改变其余元素顺序的情况下,通过删除一些元素或不删除任何元素而从 arr 派生出来的序列。
示例 1:
输入:arr = [1, 2, 3, 4], difference = 1
输出:4
解释:最长的等差子序列是[1, 2, 3, 4]。
示例 2:
输入:arr = [1, 3, 5, 7], difference = 1
输出:1
解释:最长的等差子序列是任意单个元素。
示例 3:
输入:arr = [1, 5, 7, 8, 5, 3, 4, 2, 1], difference = -2
输出:4
解释:最长的等差子序列是[7, 5, 3, 1]。
提示:
思路:
因此,这里可以选择使用哈希表做优化。我们可以把「元素, dp[j] 」绑定,放进哈希表中。甚至不用创建 dp 数组,直接在哈希表中做动态规划。
代码如下:
class Solution { public: int longestSubsequence(vector& arr, int difference) { // dp[i] 表示:以 i 位置的元素为结尾所有的⼦序列中,最长的等差子序列的长度 unordered_map hash; // arr[i], dp[i] hash[arr[0]] = 1; int n = arr.size(); int ans = 1; for(int i = 1; i < n; i++) { hash[arr[i]] = hash[arr[i] - difference] + 1; ans = max(hash[arr[i]], ans); } return ans; } };
题目链接 -> Leetcode -873.最长的斐波那契子序列的长度
Leetcode -873.最长的斐波那契子序列的长度
题目:如果序列 X_1, X_2, …, X_n 满足下列条件,就说它是 斐波那契式 的:
给定一个严格递增的正整数数组形成序列 arr ,找到 arr 中最长的斐波那契式的子序列的长度。如果一个不存在,返回 0 。
(回想一下,子序列是从原序列 arr 中派生出来的,它从 arr 中删掉任意数量的元素(也可以不删),而不改变其余元素的顺序。例如,[3, 5, 8] 是[3, 4, 5, 6, 7, 8] 的一个子序列)
示例 1:
输入 : arr = [1, 2, 3, 4, 5, 6, 7, 8]
输出 : 5
解释 : 最长的斐波那契式子序列为[1, 2, 3, 5, 8] 。
示例 2:
输入 : arr = [1, 3, 7, 11, 12, 14, 18]
输出 : 3
解释 : 最长的斐波那契式子序列有[1, 11, 12]、[3, 11, 14] 以及[7, 11, 18] 。
提示:
思路:
但是这里有一个非常致命的问题,那就是我们无法确定 i 结尾的斐波那契序列的样子。这样就会导致我们无法推导状态转移方程,因此我们定义的状态表示需要能够确定一个斐波那契序列。
根据斐波那契数列的特性,我们仅需知道序列里面的最后两个元素,就可以确定这个序列的样子。因此,我们修改我们的状态表示为:
综上,状态转移方程分情况讨论即可。
优化点:我们发现,在状态转移方程中,我们需要确定 a 元素的下标。因此我们可以在 dp 之前,将所有的「元素 + 下标」绑定在一起,放到哈希表中。
代码如下:
class Solution { public: int lenLongestFibSubseq(vector& arr) { int n = arr.size(); unordered_map hash; // arr[i] - i ,将数组中的元素与下标绑定放入哈希表中 for (int i = 0; i < n; i++) hash[arr[i]] = i; vector > dp(n, vector (n, 2)); int ans = 2; // dp[j][i] 表示以 j 和 i 位置的元素为结尾的所有子序列中,最长的斐波那契子序列的长度 for (int i = 2; i < n; i++) // 固定 i 在后 { for (int j = i - 1; j >= 1; j--) //固定 j 在前 { int tp = arr[i] - arr[j]; if (tp < arr[j] && hash.count(tp)) dp[j][i] = dp[hash[tp]][j] + 1; ans = max(dp[j][i], ans); } hash[arr[i]] = i; } return ans < 3 ? 0 : ans; } };
题目链接 -> Leetcode -1027.最长等差数列
Leetcode -1027.最长等差数列
题目:给你一个整数数组 nums,返回 nums 中最长等差子序列的长度。
回想一下,nums 的子序列是一个列表 nums[i1], nums[i2], …, nums[ik] ,且 0 <= i1 < i2 < … < ik <= nums.length - 1。
并且如果 seq[i + 1] - seq[i](0 <= i < seq.length - 1) 的值都相同,那么序列 seq 是等差的。
示例 1:
输入:nums = [3, 6, 9, 12]
输出:4
解释:
整个数组是公差为 3 的等差数列。
示例 2:
输入:nums = [9, 4, 7, 2, 10]
输出:3
解释:
最长的等差子序列是[4, 7, 10]。
示例 3:
输入:nums = [20, 1, 15, 3, 10, 5, 8]
输出:4
解释:
最长的等差子序列是[20, 15, 10, 5]。
提示:
思路:本题思路与上题思路类似,只是在推前一个元素的方法不一样,其中本题:设 nums[i] = b, nums[j] = c ,那么这个序列的前一个元素就是 a = 2 * b - c ;接下来我们和上题根据 a 的情况讨论,这里就不再做讨论。
代码如下:
class Solution { public: int longestArithSeqLength(vector& nums) { int n = nums.size(); unordered_map hash; // nums[i], i //for(int i = 0; i < n; i++) hash[nums[i]] = i; hash[nums[0]] = 0; int ans = 2; vector > dp(n, vector (n, 2)); // dp[i][j] 表示:以 i 位置以及 j 位置的元素为结尾的所有的⼦序列中,最长的等差序列的⻓度。规定 i < j for(int i = 1; i < n; i++) // 固定倒数第二个数 { for(int j = i + 1; j < n; j++) // 固定倒数第一个数 { // 求出以 i 和 j 位置结尾的子序列中,以它们为等差数列的前一个数 int tp = 2 * nums[i] - nums[j]; if(hash.count(tp)) { dp[i][j] = dp[hash[tp]][i] + 1; } ans = max(ans, dp[i][j]); } // 一边做 dp 一边更新hash数组;以便多个 tp 元素重复时,hash[tp]找到的是离 i 最近的下标 hash[nums[i]] = i; } return ans; } };
题目链接 -> 等差数列划分Ⅱ - 子序列
Leetcode -446.等差数列划分Ⅱ - 子序列
题目:给你一个整数数组 nums ,返回 nums 中所有 等差子序列 的数目。
如果一个序列中 至少有三个元素 ,并且任意两个相邻元素之差相同,则称该序列为等差序列。
例如,[1, 3, 5, 7, 9]、[7, 7, 7, 7] 和[3, -1, -5, -9] 都是等差序列。
再例如,[1, 1, 2, 5, 7] 不是等差序列。
数组中的子序列是从数组中删除一些元素(也可能不删除)得到的一个序列。
例如,[2, 5, 10] 是[1, 2, 1, 2, 4, 1, 5, 10] 的一个子序列。
题目数据保证答案是一个 32 - bit 整数。
示例 1:
输入:nums = [2, 4, 6, 8, 10]
输出:7
解释:所有的等差子序列为:
[2, 4, 6]
[4, 6, 8]
[6, 8, 10]
[2, 4, 6, 8]
[4, 6, 8, 10]
[2, 4, 6, 8, 10]
[2, 6, 10]
示例 2:
输入:nums = [7, 7, 7, 7, 7]
输出:16
解释:数组中的任意子序列都是等差子序列。
提示:
思路:
本题与最长等差数列的区别是,最长等差数列求的是最长等差数列的长度,本题求的是等差数列的数目;所以思路我们也是类似的:
状态表示:dp[i][j] 表示:以 i 位置以及 j 位置的元素为结尾的所有的子序列中,等差子序列的个数。规定⼀下 i < j.
状态转移方程:设 nums[i] = b, nums[j] = c ,那么这个序列的前一个元素就是 a = 2 * b - c 。我们根据 a 的情况讨论:
综上, dp[i][j] += dp[k][i] + 1.
代码如下:
class Solution { public: int numberOfArithmeticSlices(vector& nums) { int n = nums.size(); // 将数组的元素和下标数组绑定在一起放入哈希表中,因为相同的元素会有不同的下标 unordered_map > hash; for(int i = 0; i < n; i++) hash[nums[i]].push_back(i); vector > dp(n, vector (n)); int count = 0; for(int i = 1; i < n; i++) // 枚举倒数第二个数 { for(int j = i + 1; j < n; j++) // 枚举倒数第一个数 { long long tp = (long long)2 * nums[i] - nums[j]; if(hash.count(tp)) { // 遍历 tp 元素的下标数组,tp 的下标需要小于 i for(auto& k : hash[tp]) { // 需要 += ,是因为要确定以 i、j 为结尾的元素前面的等差数列个数,就要用以 k、i 为结尾的元素前面的等差数列个数再加上1;因为以 k、i 为结尾的元素前面的等差数列再加上 j 的元素,也能构成数量相同的等差数列个数,而且多了 k、i、j 这一组等差数列,所以要多加1 if(k < i) dp[i][j] += dp[k][i] + 1; } } count += dp[i][j]; } } return count; } };