[动态规划]Leetcode 1143.最长公共子序列(python)

简介: [动态规划]Leetcode 1143.最长公共子序列(python)

[动态规划]Leetcode 1143.最长公共子序列(python)


题目描述


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


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

例如,“ace” 是 “abcde” 的子序列,但 “aec” 不是 “abcde” 的子序列。两个字符串的「公共子序列」是这两个字符串所共同拥有的子序列。


若这两个字符串没有公共子序列,则返回 0。


示例1


输入:text1 = “abcde”, text2 = “ace”

输出:3

解释:最长公共子序列是 “ace”,它的长度为 3。


示例2


输入:text1 = “abc”, text2 = “abc”

输出:3

解释:最长公共子序列是 “abc”,它的长度为 3。


提示:


1 <= text1.length <= 1000

1 <= text2.length <= 1000

输入的字符串只含有小写英文字符。


DP定义及状态方程


一般来说对于两个字符串类型的长度问题,都会定义一个dp[i][j]代表字符串s1[1...i]s2[1...j]所求的问题。对于本题同样定义

dp[i][j]表示s1[1...i]s2[1...j]的最长公共子序列为dp[i][j]



递推方程


  1. 那么当s1[i-1]==s2[i-1]时,dp[i][j]的最长公共子序列是在dp[i-1][j-1]的基础上加1,即dp[i][j]=dp[i-1][j-1]+1;


  1. 当s1[i-1]!=s2[i-1]时,dp[i][j]的最长公共子序列为max(dp[i][j-1], d[i-1][j])


  1. 本题的目标值即为dp[len(s1)-1][len(s2)-1]


dp数组结构 如下图所示:


20201202100325919.png


初始边界条件


对于s1[0]对应的s2[j]均为0,即dp[0][j]=00=< j< len(s2);

同理 dp[i][0]=0,0 =< i < len(s1)


最终代码如下


# 动态规划,dp数组形式求解
class Solution:
    def longestCommonSubsequence(self, text1: str, text2: str) -> int:
        m = len(text1)
        n = len(text2)
        dp = [[0] * (n + 1) for _ in range(m + 1)]
        for i in range(1, m+1):
            for j in range(1, n+1):
                if text1[i-1] == text2[j-1]:
                    dp[i][j] = dp[i-1][j-1] + 1
                else:
                    dp[i][j] = max(dp[i][j-1],dp[i-1][j])
        return dp[-1][-1] #dp的最后一个数即为本题答案


附上递归形式的写法,思想其实与dp数组求解形式相同,只是需要用备忘录memo记录已经求解过的值,防止重复计算。


#递归的形式求解
class Solution:
    def longestCommonSubsequence(self, text1: str, text2: str) -> int:
        # 递归
        memo = {} #备忘录
        def dp(i, j):
            if i == -1 or j == -1:
                return 0
            if (i,j) in memo:
                return memo[(i,j)]
            if text1[i] == text2[j]:
                memo[(i,j)] = dp(i-1, j-1) + 1
            else:
                memo[(i,j)] = max(dp(i-1, j), dp(i,j-1))
            return memo[(i,j)]
        return dp(len(text1)-1,len(text2)-1)
相关文章
|
6月前
|
Python
【Leetcode刷题Python】剑指 Offer 32 - III. 从上到下打印二叉树 III
本文介绍了两种Python实现方法,用于按照之字形顺序打印二叉树的层次遍历结果,实现了在奇数层正序、偶数层反序打印节点的功能。
78 6
|
6月前
|
搜索推荐 索引 Python
【Leetcode刷题Python】牛客. 数组中未出现的最小正整数
本文介绍了牛客网题目"数组中未出现的最小正整数"的解法,提供了一种满足O(n)时间复杂度和O(1)空间复杂度要求的原地排序算法,并给出了Python实现代码。
151 2
|
3月前
|
算法 Python
在Python编程中,分治法、贪心算法和动态规划是三种重要的算法。分治法通过将大问题分解为小问题,递归解决后合并结果
在Python编程中,分治法、贪心算法和动态规划是三种重要的算法。分治法通过将大问题分解为小问题,递归解决后合并结果;贪心算法在每一步选择局部最优解,追求全局最优;动态规划通过保存子问题的解,避免重复计算,确保全局最优。这三种算法各具特色,适用于不同类型的问题,合理选择能显著提升编程效率。
87 2
|
4月前
|
存储 算法 Python
【10月更文挑战第16天】「Mac上学Python 27」小学奥数篇13 - 动态规划入门
本篇将通过 Python 和 Cangjie 双语介绍动态规划的基本概念,并解决一个经典问题:斐波那契数列。学生将学习如何使用动态规划优化递归计算,并掌握编程中的重要算法思想。
131 3
|
6月前
|
算法 Python
【Leetcode刷题Python】 LeetCode 2038. 如果相邻两个颜色均相同则删除当前颜色
本文介绍了LeetCode 2038题的解法,题目要求在一个由'A'和'B'组成的字符串中,按照特定规则轮流删除颜色片段,判断Alice是否能够获胜,并提供了Python的实现代码。
71 3
|
6月前
|
算法 Python
【Leetcode刷题Python】剑指 Offer 33. 二叉搜索树的后序遍历序列
本文提供了一种Python算法,用以判断给定整数数组是否为某二叉搜索树的后序遍历结果,通过识别根节点并递归验证左右子树的值是否满足二叉搜索树的性质。
34 3
|
6月前
|
Python
【Leetcode刷题Python】剑指 Offer 32 - II. 从上到下打印二叉树 II
本文提供了一种Python实现方法,用于层次遍历二叉树并按层打印结果,每层节点按从左到右的顺序排列,每层打印到一行。
51 3
|
6月前
|
Python
【Leetcode刷题Python】剑指 Offer 32 - I. 从上到下打印二叉树
本文介绍了使用Python实现从上到下打印二叉树的解决方案,采用层次遍历的方法,利用队列进行节点的访问。
48 2
|
6月前
|
算法 Python
【Leetcode刷题Python】73. 矩阵置零
本文介绍了LeetCode第73题的解法,题目要求在给定矩阵中将所有值为0的元素所在的行和列全部置为0,并提供了一种原地算法的Python实现。
56 0
【Leetcode刷题Python】73. 矩阵置零
|
人工智能 算法 Python
「只出现一次的数字」python之leetcode刷题|007
题目 给定一个非空整数数组,除了某个元素只出现一次以外,其余每个元素均出现了三次。找出那个只出现了一次的元素。 说明: 你的算法应该具有线性时间复杂度。
1506 0

热门文章

最新文章