代码随想录笔记_动态规划
代码随想录二刷笔记记录
LC1035.不相交的线
题目
子序列问题
在两条独立的水平线上按给定的顺序写下 nums1 和 nums2 中的整数。
现在,可以绘制一些连接两个数字 nums1[i] 和 nums2[j] 的直线,这些直线需要同时满足满足:
- nums1[i] == nums2[j]
- 且绘制的直线不与任何其他连线(非水平线)相交。
请注意,连线即使在端点也不能相交:每个数字只能属于一条连线。
以这种方法绘制线条,并返回可以绘制的最大连线数。
示例 1:
输入:nums1 = [1,4,2], nums2 = [1,2,4]
输出:2
解释:可以画出两条不交叉的线,如上图所示。
但无法画出第三条不相交的直线,因为从 nums1[1]=4 到 nums2[2]=4 的直线将与从 nums1[2]=2 到 nums2[1]=2 的直线相交。
示例 2:
输入:nums1 = [2,5,1,2,5], nums2 = [10,5,2,1,5,2]
输出:3
示例 3:
输入:nums1 = [1,3,7,1,7,5], nums2 = [1,9,2,5,1]
输出:2
思路分析
思路:
以 A = [1,4,2], B = [1,2,4] 为例,我们可知,在A中 [1,4] 为一个子序列。在 B 中,也需保持 1 - … - 4 的相对顺序。 1和4之中包含几个数字不重要。只要保持1-4的相对顺序,则A中的1-4,就可以和B连线。
也可以理解为,A的子序列,在B中存在。则转化为寻找子序列的问题。
1 4 2
| \
1 2 4
根据5.推演,我们可知,本题和 LC1143 是一样的。区别在于,本题是数组,而 LC1143是字符串。
动态规划五部曲
1.确定dp数组及其下标的含义
dp[i][j] : 以 i-1 结尾的 nums1数组的子序列,和以 j-1 结尾的 num2 数组的子序列的最长长度。
这里和 lc1143 一样,需要在行列增维,方便计算。
2.确定递推公式
当 A[i-1] 和 B[j-1] 相等时,我们则需要更新 dp[i][j],因此有
dp[i][j] = dp[i-1][j-1] + 1
当 A[i-1] 和 B[j-1] 不相等时,从A[i-1,j-2],B[i-2,j-1]中取目前最长子序列的长度
dp[i][j] = Max(dp[i-1][j],dp[i][j-1])
3.初始化
由于增维,因此 dp[i][0],dp[0][j] 没有意义
所以我们需要将其初始化为 0
4.遍历顺序
根据递推公式可知,当前状态的推导需要前一个状态。因此是从前往后遍历。
for (int i = 1; i <= len1; i++) {
for (int j = 1; j <= len2; j++) {
if (nums1[i-1] == nums2[j-1]){
dp[i][j] = dp[i-1][j-1] + 1;
}else {
dp[i][j] = Math.max(dp[i - 1][j],dp[i][j - 1]);
}
}
}
5.推演分析
以 A = [1,4,2], B = [1,2,4] 为例
B(j) | 1 | 2 | 4 | |
---|---|---|---|---|
A(i) | 0 | 0 | 0 | 0 |
1 | 0 | 1 | 1 | 1 |
4 | 0 | 1 | 1 | 2 |
2 | 0 | 1 | 2 | 2 |
代码实现
完整代码实现
public int maxUncrossedLines(int[] nums1, int[] nums2) {
int len1 = nums1.length;
int len2 = nums2.length;
int[][] dp = new int[len1 + 1][len2 + 1];
//遍历
for (int i = 1; i <= len1; i++) {
for (int j = 1; j <= len2; j++) {
if (nums1[i-1] == nums2[j-1]){
dp[i][j] = dp[i-1][j-1] + 1;
}else {
dp[i][j] = Math.max(dp[i - 1][j],dp[i][j - 1]);
}
}
}
return dp[len1][len2];
}