力扣 095. 最长公共子序列(C语言+动态规划)

2023-10-15 00:52

本文主要是介绍力扣 095. 最长公共子序列(C语言+动态规划),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1. 题目

        给定两个字符串 text1 和 text2,返回这两个字符串的最长 公共子序列 的长度。如果不存在 公共子序列 ,返回 0 。

        一个字符串的 子序列 是指这样一个新的字符串:它是由原字符串在不改变字符的相对顺序的情况下删除某些字符(也可以不删除任何字符)后组成的新字符串。

        例如,"ace" 是 "abcde" 的子序列,但 "aec" 不是 "abcde" 的子序列。

        两个字符串的 公共子序列 是这两个字符串所共同拥有的子序列。

2. 输入输出样例

        示例 1:

输入:text1 = "abcde", text2 = "ace" 
输出:3  
解释:最长公共子序列是 "ace" ,它的长度为 3 

        示例 2: 

输入:text1 = "abc", text2 = "abc"
输出:3
解释:最长公共子序列是 "abc" ,它的长度为 3 

         示例 3:

输入:text1 = "abc", text2 = "def"
输出:0
解释:两个字符串没有公共子序列,返回 0 。

 提示:

  • 1 <= text1.length, text2.length <= 1000
  • text1 和 text2 仅由小写英文字符组成。

3. 解题思想

        动态规划步骤:

        (1)dp状态:

                dp[i][j]表示以text1[i]、text2[j]为结尾的两个字符串中最长公共子序列的长度;

        (2)状态转移方程:

                text1[i] == text2[j]:dp[i][j] = dp[i - 1][j - 1] + 1;

                text1[i] != text2[j]:max(dp[i - 1][j], dp[i][j - 1]);

        (3)初始化状态:

                第0行第0列:text1[0] == text2[0]:dp[0][0] = 1;text1[0] != text2[0]:dp[0][0] = 0;

                第0行:text1[i] == text2[0]:dp[i][0] = 1;text1[i] != text2[0]:dp[i][0] = dp[i - 1][0];

                第0列:text1[0] == text2[i]:dp[0][1] = 1;text1[0] != text2[i]:dp[0][i] = dp[0][i-1];

         (4)最优解:

                dp[n-1][m-1] ;

        算法描述:

        核心思想是通过填充 dp 数组,逐步构建最长公共子序列的长度,考虑字符是否匹配。

  • 首先,获取输入字符串 text1text2 的长度,并创建一个二维数组 dp,其大小为 (n+1) x (m+1),其中 nm 分别是两个字符串的长度。dp[i][j] 表示 text1 的前 i 个字符和 text2 的前 j 个字符的最长公共子序列的长度。
  • 初始化 dp 数组的第一行和第一列:遍历两个字符串的首字符,如果它们相等,将 dp[0][0] 设置为1,否则将其保留为0。接着,初始化第一行和第一列的其余部分,以表示以 text1[0]text2[0] 开头的子序列。
  • 使用两个嵌套循环遍历 text1text2 的每个字符(除去第一个字符),填充 dp 数组。如果当前字符相同(text1[i] == text2[j]),则将 dp[i][j] 设置为左上角的对角元素值加1,表示找到了一个更长的公共子序列。如果当前字符不同,将 dp[i][j] 设置为左边或上边的较大值,表示要么继承左边的最长子序列长度,要么继承上边的最长子序列长度。
  • 最终,dp[n-1][m-1] 中存储的值即为 text1text2 的最长公共子序列的长度。

4. 代码实现

// 定义一个函数,该函数返回两个整数指针中的较大值
int max_(int *a, int *b) {// 比较两个指针的值,返回较大的指针if (a > b) {return a;}return b;
}// 定义一个计算两个字符串的最长公共子序列的函数
int longestCommonSubsequence(char *text1, char *text2) {// 获取字符串text1和text2的长度int n = strlen(text1);int m = strlen(text2);// 创建一个二维数组dp,用于存储最长公共子序列的长度int dp[n][m];// 初始化dp数组,将所有元素设置为0for (int i = 0; i < n; i++) {for (int j = 0; j < m; j++) {dp[i][j] = 0;}}// 初始化dp数组的第一个元素if (text1[0] == text2[0]) {dp[0][0] = 1;}// 处理第一列,初始化以text1[0]为开头的子序列for (int i = 1; i < n; i++) {if (text1[i] == text2[0]) {dp[i][0] = 1;} else {dp[i][0] = dp[i - 1][0];}}// 处理第一行,初始化以text2[0]为开头的子序列for (int i = 1; i < m; i++) {if (text1[0] == text2[i]) {dp[0][i] = 1;} else {dp[0][i] = dp[0][i - 1];}}// 填充dp数组的其余部分,找到最长公共子序列的长度for (int i = 1; i < n; i++) {for (int j = 1; j < m; j++) {if (text1[i] == text2[j]) {// 如果字符相同,将dp[i][j]设置为左上角值加1dp[i][j] = dp[i - 1][j - 1] + 1;} else {// 如果字符不相同,将dp[i][j]设置为左边和上边的较大值dp[i][j] = max_(dp[i - 1][j], dp[i][j - 1]);}}}// 返回dp数组的最右下角元素,即最长公共子序列的长度return dp[n - 1][m - 1];
}

 5. 复杂度分析

        时间复杂度分析:

  • 初始化 dp 数组的两个嵌套循环(for 循环嵌套)需要遍历整个数组,时间复杂度为O(n * m),其中 n 和 m 分别是 text1text2 的长度。
  • 接下来,还需要一个嵌套循环来填充 dp 数组,这个循环也需要遍历整个 dp 数组,时间复杂度为O(n * m)。
  • 总的时间复杂度是O(n * m + n * m),即O(n * m)。

        算法的时间复杂度是 O(n * m),其中 n 和 m 分别是输入字符串 text1text2 的长度。

        

        空间复杂度分析:

  • dp 数组的空间复杂度是O(n * m),因为它是一个二维数组,其大小与输入字符串的长度相关。

综上所述,这段代码的空间复杂度是 O(n * m)时间复杂度是 O(n * m)

 

 力扣(LeetCode)官网 - 全球极客挚爱的技术成长平台https://leetcode.cn/problems/qJnOS7/submissions/

 

这篇关于力扣 095. 最长公共子序列(C语言+动态规划)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/214301

相关文章

C语言中%zu的用法解读

《C语言中%zu的用法解读》size_t是无符号整数类型,用于表示对象大小或内存操作结果,%zu是C99标准中专为size_t设计的printf占位符,避免因类型不匹配导致错误,使用%u或%d可能引发... 目录size_t 类型与 %zu 占位符%zu 的用途替代占位符的风险兼容性说明其他相关占位符验证示

C语言进阶(预处理命令详解)

《C语言进阶(预处理命令详解)》文章讲解了宏定义规范、头文件包含方式及条件编译应用,强调带参宏需加括号避免计算错误,头文件应声明函数原型以便主函数调用,条件编译通过宏定义控制代码编译,适用于测试与模块... 目录1.宏定义1.1不带参宏1.2带参宏2.头文件的包含2.1头文件中的内容2.2工程结构3.条件编

go动态限制并发数量的实现示例

《go动态限制并发数量的实现示例》本文主要介绍了Go并发控制方法,通过带缓冲通道和第三方库实现并发数量限制,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面... 目录带有缓冲大小的通道使用第三方库其他控制并发的方法因为go从语言层面支持并发,所以面试百分百会问到

Go语言并发之通知退出机制的实现

《Go语言并发之通知退出机制的实现》本文主要介绍了Go语言并发之通知退出机制的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录1、通知退出机制1.1 进程/main函数退出1.2 通过channel退出1.3 通过cont

Go语言编译环境设置教程

《Go语言编译环境设置教程》Go语言支持高并发(goroutine)、自动垃圾回收,编译为跨平台二进制文件,云原生兼容且社区活跃,开发便捷,内置测试与vet工具辅助检测错误,依赖模块化管理,提升开发效... 目录Go语言优势下载 Go  配置编译环境配置 GOPROXYIDE 设置(VS Code)一些基本

Spring的RedisTemplate的json反序列泛型丢失问题解决

《Spring的RedisTemplate的json反序列泛型丢失问题解决》本文主要介绍了SpringRedisTemplate中使用JSON序列化时泛型信息丢失的问题及其提出三种解决方案,可以根据性... 目录背景解决方案方案一方案二方案三总结背景在使用RedisTemplate操作redis时我们针对

深入理解Go语言中二维切片的使用

《深入理解Go语言中二维切片的使用》本文深入讲解了Go语言中二维切片的概念与应用,用于表示矩阵、表格等二维数据结构,文中通过示例代码介绍的非常详细,需要的朋友们下面随着小编来一起学习学习吧... 目录引言二维切片的基本概念定义创建二维切片二维切片的操作访问元素修改元素遍历二维切片二维切片的动态调整追加行动态

一文详解SpringBoot中控制器的动态注册与卸载

《一文详解SpringBoot中控制器的动态注册与卸载》在项目开发中,通过动态注册和卸载控制器功能,可以根据业务场景和项目需要实现功能的动态增加、删除,提高系统的灵活性和可扩展性,下面我们就来看看Sp... 目录项目结构1. 创建 Spring Boot 启动类2. 创建一个测试控制器3. 创建动态控制器注

Go语言中make和new的区别及说明

《Go语言中make和new的区别及说明》:本文主要介绍Go语言中make和new的区别及说明,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1 概述2 new 函数2.1 功能2.2 语法2.3 初始化案例3 make 函数3.1 功能3.2 语法3.3 初始化

Go语言中nil判断的注意事项(最新推荐)

《Go语言中nil判断的注意事项(最新推荐)》本文给大家介绍Go语言中nil判断的注意事项,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录1.接口变量的特殊行为2.nil的合法类型3.nil值的实用行为4.自定义类型与nil5.反射判断nil6.函数返回的