代码学习第43天---动态规划

随想录日记part43

t i m e : time: time: 2024.04.15



主要内容:今天开始要学习动态规划的相关知识了,今天的内容主要涉及:判断子序列;不同的子序列



动态规划五部曲:

【1】.确定dp数组以及下标的含义

【2】.确定递推公式

【3】.dp数组如何初始化

【4】.确定遍历顺序

【5】.举例推导dp数组

Topic1判断子序列

题目:

思路:

接下来进行动规五步曲:

1.确定dp数组以及下标的含义:

dp[i][j]:长度为[0, i - 1]的s与长度为[0, j - 1]的t的最长公共子序列为dp[i][j]

2.确定递推公式:

主要就是两大情况:1.s[i - 1] 与 t[j - 1]相同 2.s[i - 1] 与 t[j - 1]不相同

如果s[i - 1] 与 t[j - 1]相同,那么找到了一个公共元素,所以dp[i][j] = dp[i - 1][j - 1] + 1;

如果s[i - 1] 与 t[j - 1]不相同,dp[i][j] = dp[i][j - 1];

3.dp数组如何初始化

java 复制代码
 for(int i=0;i<=nums1.length;i++) dp[i][0]=0;
 for(int i=0;i<=nums2.length;i++) dp[0][i]=0;

4.确定遍历顺序

外层for循环遍历A,内层for循环遍历B。

5.举例推导dp数组

以输入:text1 = "abcde", text2 = "ace" 为例,dp状态如图:

java 复制代码
class Solution {
    public boolean isSubsequence(String s, String t) {
        int n = s.length();
        int m = t.length();
        if (n > m)
            return false;
        if (n == 0)
            return true;
        // dp[i][j] 表示长度为i的序列是否是长度为j的序列的公共最长子序列的长度;
        int[][] dp = new int[n + 1][m + 1];
        // 初始化
        for (int i = 1; i <= n; i++) {
            for (int j = 1; j <= m; j++) {
                if (s.charAt(i - 1) == t.charAt(j - 1))
                    dp[i][j] = dp[i - 1][j - 1] + 1;
                else
                    dp[i][j] = dp[i][j - 1];
            }
        }
        if (dp[n][m] == n)
            return true;
        else
            return false;

    }
}

时间复杂度 : O ( n ∗ m ) O(n*m) O(n∗m)
空间复杂度 : O ( n ∗ m ) O(n*m) O(n∗m)



Topic2不同的子序列

思路:

接下来进行动规五步曲:

1.确定dp数组以及下标的含义:

dp[i][j]:以i-1为结尾的s子序列中出现以j-1为结尾的t的个数为dp[i][j]

2.确定递推公式:

主要就是两大情况:1.s[i - 1] 与 t[j - 1]相同 2.s[i - 1] 与 t[j - 1]不相同

如果s[i - 1] 与 t[j - 1]相同,dp[i][j] = dp[i - 1][j - 1] + dp[i][j - 1];

如果s[i - 1] 与 t[j - 1]不相同,dp[i][j] = dp[i][j - 1];

3.dp数组如何初始化

java 复制代码
 for(int i=0;i<=nums1.length;i++) dp[i][0]=1;
 for(int i=0;i<=nums2.length;i++) dp[0][i]=0;

4.确定遍历顺序

外层for循环遍历A,内层for循环遍历B。

5.举例推导dp数组

以s:"baegg",t:"bag"为例,推导dp数组状态如下:

java 复制代码
class Solution {
    public int numDistinct(String s, String t) {
        int n = s.length();
        int m = t.length();
        if (m > n)
            return 0;
        int[][] dp = new int[n + 1][m + 1];
        for (int i = 0; i < n + 1; i++)
            dp[i][0] = 1;
        for (int i = 1; i < n + 1; i++) {
            for (int j = 1; j < m + 1; j++) {
                if (s.charAt(i - 1) == t.charAt(j - 1)) {
                    dp[i][j] = (dp[i - 1][j - 1] + dp[i - 1][j]) % ((int) 1e9 + 7);
                } else {
                    dp[i][j] = dp[i - 1][j] % ((int) 1e9 + 7);
                }
            }
        }
        return dp[n][m];
    }
}

时间复杂度 : O ( n ∗ m ) O(n*m) O(n∗m)
空间复杂度 : O ( n ∗ m ) O(n*m) O(n∗m)

相关推荐
算AI8 小时前
人工智能+牙科:临床应用中的几个问题
人工智能·算法
云上艺旅9 小时前
K8S学习之基础七十四:部署在线书店bookinfo
学习·云原生·容器·kubernetes
你觉得2059 小时前
哈尔滨工业大学DeepSeek公开课:探索大模型原理、技术与应用从GPT到DeepSeek|附视频与讲义下载方法
大数据·人工智能·python·gpt·学习·机器学习·aigc
hyshhhh10 小时前
【算法岗面试题】深度学习中如何防止过拟合?
网络·人工智能·深度学习·神经网络·算法·计算机视觉
A旧城以西10 小时前
数据结构(JAVA)单向,双向链表
java·开发语言·数据结构·学习·链表·intellij-idea·idea
无所谓จุ๊บ10 小时前
VTK知识学习(50)- 交互与Widget(一)
学习·vtk
FAREWELL0007510 小时前
C#核心学习(七)面向对象--封装(6)C#中的拓展方法与运算符重载: 让代码更“聪明”的魔法
学习·c#·面向对象·运算符重载·oop·拓展方法
杉之10 小时前
选择排序笔记
java·算法·排序算法
吴梓穆11 小时前
UE5学习笔记 FPS游戏制作38 继承标准UI
笔记·学习·ue5
烂蜻蜓11 小时前
C 语言中的递归:概念、应用与实例解析
c语言·数据结构·算法