【动态规划】最长公共子序列
最长公共子序列
最长公共子序列(Longest Common Subsequence),英文缩写为LCS,是动态规划中的经典问题。其定义是,一个序列 S,如果分别是两个或多个已知序列的子序列,且是所有符合此条件序列中最长的,则 S 称为已知序列的最长公共子序列。
典型应用如下:
序号 | 题目 |
---|---|
1 | 1143. 最长公共子序列 |
2 | 583. 两个字符串的删除操作 |
应用
应用1:Leetcode.1143
题目
解题思路
我们使用动态规划求解,设
即子串
初始状态
当其中一个字符串的长度为
状态转移
我们分别枚举两个字符串,会出现两种情况:
- 如果当前两个字符相等,那么当前状态
就可以由上一个状态 加 ; - 如果当前两个字符不相等,那么当前状态
就要从 或者 中取最大值。
所以,状态转移方程如下:
代码
class Solution: def longestCommonSubsequence(self, text1: str, text2: str) -> int: m, n = len(text1), len(text2) dp = [[0 for _ in range(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[m][n]
应用2:Leetcode.583
题目
解题思路
先找到两个字符串的最长公共子序列,然后再用两个字符串的长度分别减去最长公共子序列长度,剩下的就是不相等的字符,即要删除字符。
算法的思路:
- 先求两个字符串的最长公共子序列;
- 需要删除的字符,就是公共子序列;
代码实现
class Solution: def minDistance(self, word1: str, word2: str) -> int: m, n = len(word1), len(word2) length = self.lcs(word1, word2) return m - length + n - length def lcs(self, text1: str, text2: str): m, n = len(text1), len(text2) dp = [[0 for _ in range(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[m][n]
总结
动态规划的本质就是:枚举所有的状态,并寻找最优解。
本文作者:LARRY1024
本文链接:https://www.cnblogs.com/larry1024/p/16891961.html
版权声明:本作品采用知识共享署名-非商业性使用-禁止演绎 2.5 中国大陆许可协议进行许可。
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步