个人主页:元清加油_【C++】,【C语言】,【数据结构与算法】-CSDN博客
个人专栏
力扣递归题
【C++】
http://t.csdnimg.cn/6AbpV
数据结构
前言:这个专栏主要讲述动态规划算法,所以下面题目主要也是这些算法做的
我讲述题目会把讲解部分分为3个部分:
1、题目解析
2、算法原理思路讲解
3、代码实现
最长数对链
题目链接:最长数对链
题目
给你一个由 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] 。
提示:
n == pairs.length
1 <= n <= 1000
-1000 <= lefti < righti <= 1000
解法
算法原理解析
我们这题使用动态规划,我们做这类题目可以分为以下五个步骤
- 状态显示
- 状态转移方程
- 初始化(防止填表时不越界)
- 填表顺序
- 返回值
- 状态显示
dp[i]
表⽰以
i
位置的数对为结尾时,最⻓数对链的⻓度。
- 状态转移方程
对于
dp[i]
,遍历所有
[0, i - 1]
区间内数对⽤
j
表⽰下标,找出所有满⾜
pairs[j] [1] < pairs[i][0] 的
j
。找出⾥⾯最⼤的
dp[j]
,然后加上
1
,就是以
i
位置为结尾的最⻓数对链。
- 初始化(防止填表时不越界)
刚开始的时候,全部初始化为
1。
- 填表顺序
根据「状态转移⽅程」,填表顺序应该是「从左往右」。
- 返回值
根据「状态表⽰」,返回整个
dp
表中的最⼤值。
代码实现
class Solution {
public:
int findLongestChain(vector<vector<int>>& pairs)
{
sort(pairs.begin(), pairs.end()); // 排序
int n = pairs.size();
vector<int> dp(n, 1); // 创建 dp数组,顺便初始化
int ret = 1; // 记录最⼤值
for (int i = 1; i < n; i++)
{
for (int j = 0; j < i; j++)
{
// 找出所有情况下的最⼤值
if (pairs[j][1] < pairs[i][0])
{
dp[i] = max(dp[i], dp[j] + 1);
}
}
ret = max(ret, dp[i]); // 更新最⼤值
}
return dp[n - 1];
}
};