剑指 Offer II 095. 最长公共子序列 经典DP

想说的话

大家好🌼🌼,我是 @愿此后再无WA,可以叫我小A🌀🌀,一位阳光帅小伙,对算法领域比较感兴趣。如果我的文章对您有用,欢迎持续关注,我们一起进步!🎈🎈

很抱歉各位😪😪,现离蓝桥杯仅剩一个月时间,我临时改变了计划,转为全心备战蓝桥,因为这个省一对我来说太重要了,也是我最后一次机会,我一定要拿到手📌📌,那么这样的话我在博客上花的时间就会少了很多,也将导致博客文章质量明显下降,在此我给大家说声抱歉💥💥


🌟🌟这些日子我真的很开心,博客上能遇到一群志同道合的兄弟姐妹真的很幸福,没有你们的支持与鼓励我早就坚持不下去了,因为有你们我才能走的更远☀️☀️熬过这段时期我一定会回来的,爱你们❤️❤️

题目

在这里插入图片描述

分析

这是DP很经典的一道题,做这道题需要想明白如何用DP表示。可以这么定义:在字符串str1与str2中,用一个二维DP分别表示str1与str2的公共序列长度,其中DP[i][j]表示str1[0] ~ str1[i]与str2[0] ~ str2[j]的公共子序列长度。那么DP[i][j]就会出现一下几种情况:

1、str[i] == str[j]:

如果它两相等,那么就看除它俩外前面的公共子序列长度,即str1[0i-1],str2[0j-1]。然后让前面的已经得出的子序列长度+1.

2、str[i] != str[j]:

如果它两不相等,就要看str1[0~ i-1]与str1[0~ j]的公共子序列长度,还有看str1[0~ j-1]与str1[0~i-1]的公共子序列长度了,谁长就要谁的长度。怎么理解这句话?就是有两个串,是串A的子串在串B子串的公共子序列长度长些,还是串B的子串在串A的子串公共子序列长度长些。谁长就选谁。在打表中这两者是已经得出结果了的,所以只需要选二者中的最大值即可。

代码

class Solution:
    def longestCommonSubsequence(self, text1: str, text2: str) -> int:
        N, M = len(text1), len(text2)  
        # 打表范围是 0~N,0~M。
        dp = [[0 for i in range(N+1)]for j in range(M+1)]
        for i in range(1,M+1):
            for j in range(1,N+1):
                # 如果相等,就是之前得到的最大公共子序列+1
                if text1[j-1] == text2[i-1]:
                    dp[i][j] = dp[i-1][j-1]+1
                else:
                    # 要么是串A子串与串B子串的公共序列长度,要么是串B子串与串A子串的公共序列长度,
                    # ij不同,两子串长度就不一定相同
                    dp[i][j] = max(dp[i-1][j],dp[i][j-1])
        return(dp[i][j])
  • 4
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

愿此后再无WA

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值