代码随想录笔记_动态规划_1035不相交的线

代码随想录笔记_动态规划

代码随想录二刷笔记记录

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)124
A(i)0000
10111
40112
20122

代码实现

完整代码实现

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];
    }

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值