当前位置: 首页 > article >正文

代码随想录算法训练营第四十五天|Day45 动态规划

115.不同的子序列

https://programmercarl.com/0115.%E4%B8%8D%E5%90%8C%E7%9A%84%E5%AD%90%E5%BA%8F%E5%88%97.html

思路

#define MOD 1000000007
int numDistinct(char* s, char* t) {
    int m = strlen(s);
    int n = strlen(t);
    int dp[m + 1][n + 1];
    memset(dp, 0, sizeof(dp));
    dp[0][0] = 1; 
    for (int i = 0; i <= m; i++) {
        dp[i][0] = 1; 
    }
    for (int j = 1; j <= n; j++) {
        dp[0][j] = 0; 
    for (int i = 1; i <= m; i++) {
        for (int j = 1; j <= n; j++) {
            if (s[i - 1] == t[j - 1]) {
                dp[i][j] = (dp[i - 1][j - 1] + dp[i - 1][j]) % MOD;
            } else {
                dp[i][j] = dp[i - 1][j];
            }
        }
    }
    return dp[m][n];
}

学习反思

动态规划解决了一个字符串问题。函数numDistinct计算字符串t在字符串s中出现的次数。代码中使用了一个二维数组dp来存储状态。dp[i][j]表示s的前i个字符中包含字符串t的前j个字符的子序列数。首先,初始化dp[0][0]为1表示空字符串是空字符串的子序列。然后,初始化dp[i][0]为1表示任意非空字符串是空字符串的子序列。初始化dp[0][j]为0表示任意字符串不能包含非空字符串。接下来,使用两层循环遍历数组s和数组t的每个字符。如果s[i-1]t[j-1]相等,则有两种情况:一种是s的前i-1个字符中包含t的前j-1个字符的子序列数,另一种是s的前i-1个字符中包含t的前j个字符的子序列数。所以,dp[i][j]等于这两种情况的和。如果s[i-1]t[j-1]不相等,则dp[i][j]等于s的前i-1个字符中包含t的前j个字符的子序列数。最后,返回dp[m][n],即s包含t的子序列数。这段代码的时间复杂度是O(m*n),其中m和n分别是字符串st的长度。

583. 两个字符串的删除操作

https://programmercarl.com/0583.%E4%B8%A4%E4%B8%AA%E5%AD%97%E7%AC%A6%E4%B8%B2%E7%9A%84%E5%88%A0%E9%99%A4%E6%93%8D%E4%BD%9C.html

思路

int minDistance(char* word1, char* word2) {
    int m = strlen(word1);
    int n = strlen(word2);
    int **dp = (int **)malloc((m + 1) * sizeof(int *));
    for (int i = 0; i <= m; i++) {
        dp[i] = (int *)malloc((n + 1) * sizeof(int));
    }
    for (int i = 0; i <= m; i++) {
        dp[i][0] = i; 
    }
    for (int j = 0; j <= n; j++) {
        dp[0][j] = j; 
  }
    for (int i = 1; i <= m; i++) {
        for (int j = 1; j <= n; j++) {
            if (word1[i - 1] == word2[j - 1]) {
                dp[i][j] = dp[i - 1][j - 1]; 
            } else {
                dp[i][j] = (dp[i - 1][j] < dp[i][j - 1]) ? dp[i - 1][j] + 1 : dp[i][j - 1] + 1;
            }
        }
    }
    int result = dp[m][n];
    for (int i = 0; i <= m; i++) {
        free(dp[i]);
    }
    free(dp);
    return result;
}

学习反思

动态规划解决了一个字符串问题。函数minDistance计算将字符串word1转换成字符串word2所需的最小操作次数。代码中使用了一个二维数组dp来存储状态。dp[i][j]表示将word1的前i个字符转换成word2的前j个字符所需的最小操作次数。首先,动态申请二维数组dp的内存空间,并初始化数组。初始化dp[i][0]i,表示将word1的前i个字符转换成空字符串所需的最小操作次数。初始化dp[0][j]j,表示将空字符串转换成word2的前j个字符所需的最小操作次数。接下来,使用两层循环遍历字符串word1和字符串word2的每个字符。如果word1[i-1]word2[j-1]相等,则不需要进行任何操作,所以dp[i][j]等于dp[i-1][j-1]。如果word1[i-1]word2[j-1]不相等,则可以选择删除word1[i-1]或者插入word2[j-1],取两种操作中的较小值,并加上一次操作的代价。最后,返回dp[m][n],即将word1转换成word2所需的最小操作次数。由于申请了动态内存空间,需要在函数结束后释放内存,以避免内存泄漏。这段代码的时间复杂度是O(m*n),其中m和n分别是字符串word1word2的长度。

72. 编辑距离

https://programmercarl.com/0072.%E7%BC%96%E8%BE%91%E8%B7%9D%E7%A6%BB.html

思路

int minDistance(char* word1, char* word2) {
    int m = strlen(word1);
    int n = strlen(word2);
    int **dp = (int **)malloc((m + 1) * sizeof(int *));
    for (int i = 0; i <= m; i++) {
        dp[i] = (int *)malloc((n + 1) * sizeof(int));
    }
    for (int i = 0; i <= m; i++) {
        dp[i][0] = i;
    }    
    for (int j = 0; j <= n; j++) {
        dp[0][j] = j; 
    }
    for (int i = 1; i <= m; i++) {
        for (int j = 1; j <= n; j++) {
            if (word1[i - 1] == word2[j - 1]) {
                dp[i][j] = dp[i - 1][j - 1]; 
            } else {
                dp[i][j] = fmin(fmin(dp[i - 1][j] + 1, dp[i][j - 1] + 1), dp[i - 1][j - 1] + 1);
            }
        }
    }
    int result = dp[m][n];
    for (int i = 0; i <= m; i++) {
        free(dp[i]);
    }
    free(dp);
    return result;
}

学习反思

在计算dp[i][j]的时候,使用了fmin函数来取三者中的最小值,即dp[i - 1][j] + 1dp[i][j - 1] + 1dp[i - 1][j - 1] + 1中的最小值。这样可以简化代码并提高效率。其他部分保持不变,申请动态内存空间、初始化数组、更新dp[i][j]的值和释放内存空间的操作都没有改变。这段代码的时间复杂度仍然是O(m*n)。通过使用fmin函数,可以更清晰地表示出取三者中的最小值的操作。

总结

加油!!!


http://www.kler.cn/a/392144.html

相关文章:

  • 【llm/ollama/qwen】在本地部署qwen2.5-coder并在vscode中集成使用代码提示功能
  • 缓存-Redis-常见问题-缓存击穿-永不过期+逻辑过期(全面 易理解)
  • 网络安全-web应用程序发展历程(基础篇)
  • SpringBoot插件
  • n 维数组(张量)关于轴 axis 的理解
  • 杭州市有哪些大学能够出具论文检索报告?
  • pandas的to_sql方法中使用if_exists=‘replace‘
  • Spring Boot编程训练系统:最佳实践与技巧
  • MySQL与Oracle对比及区别
  • 图像增强——代数运算
  • vue3面试题1|[2024-11-12]
  • labview用sql server数据库存取数据到一个单元格
  • AI: 情景模拟攻击(草稿)
  • 蓝队的基础
  • 奥迪:在工业边缘使用 VMware 边缘计算堆栈
  • 从 O(n²) 到 O(n):单调栈在算法中的妙用
  • SpringSecurity Demo实操
  • 【系统架构设计师】真题论文: 论软件可靠性设计与应用(包括解题思路和素材)
  • Spring Boot编程训练系统:实战开发技巧
  • Normal-GS: 3D Gaussian Splatting with Normal-Involved Rendering 论文解读
  • 【vue】echarts地图添加蒙版图片,多图层地图实现天气信息展示
  • Hadoop生态圈框架部署(六)- HBase完全分布式部署
  • λ矩阵与矩阵的Jordan标准形
  • 蓝牙BLE开发——iOS 每次写入数据超过200字节报错?
  • CSS教程(八)- 盒子模型
  • Oracle的字符串函数