【LeetCode 热题100】字符串 DP 三连:最长回文子串、最长公共子序列 & 编辑距离(力扣5 / 1143/ )(Go语言版)

简介: 本文详细解析了字符串动态规划(DP)领域的三个经典问题:**最长回文子串**(LeetCode 5)、**最长公共子序列**(LeetCode 1143)和**编辑距离**(LeetCode 72)。通过定义状态、推导状态转移方程,结合 Go 语言实现代码,深入浅出地讲解了解题思路。从判断子串是否为回文到求解两个字符串的匹配长度,再到计算字符串转换的最小操作数,每道题都展示了 DP 的核心思想与应用场景。最后通过表格总结对比三题的异同,帮助读者更好地掌握字符串 DP 的解题技巧。

✨ 字符串 DP 三连:最长回文子串、最长公共子序列 & 编辑距离(LeetCode 5 / 1143 / 72)

  • 🔁 5. 最长回文子串
  • 📏 1143. 最长公共子序列
  • ✂️ 72. 编辑距离

1️⃣ 最长回文子串(LeetCode 5)

📌 题目描述

给你一个字符串 s,找到 s 中最长的回文子串。你可以假设 s 的最大长度为 1000。


🧠 解题思路:动态规划

✅ 状态定义

dp[i][j] 表示子串 s[i...j] 是否是回文串。

🔁 状态转移方程

  • 如果 s[i] == s[j]

    • j - i < 3,则 dp[i][j] = true
    • 否则:dp[i][j] = dp[i+1][j-1]
  • 否则:dp[i][j] = false

✅ Go 实现

func longestPalindrome(s string) string {
   
    n := len(s)
    if n < 2 {
   
        return s
    }

    dp := make([][]bool, n)
    for i := range dp {
   
        dp[i] = make([]bool, n)
    }

    maxLen := 1
    start := 0

    for r := 1; r < n; r++ {
   
        for l := 0; l < r; l++ {
   
            if s[l] == s[r] {
   
                if r - l < 3 {
   
                    dp[l][r] = true
                } else {
   
                    dp[l][r] = dp[l+1][r-1]
                }
            }
            if dp[l][r] && r - l + 1 > maxLen {
   
                maxLen = r - l + 1
                start = l
            }
        }
    }

    return s[start:start+maxLen]
}

2️⃣ 最长公共子序列(LeetCode 1143)

📌 题目描述

给定两个字符串 text1text2,返回它们的最长公共子序列的长度。


🧠 解题思路:二维 DP

✅ 状态定义

dp[i][j] 表示 text1[0...i-1]text2[0...j-1] 的最长公共子序列长度。

🔁 状态转移

  • 如果 text1[i-1] == text2[j-1],说明当前字符可加入序列:
    dp[i][j] = dp[i-1][j-1] + 1
  • 否则:
    dp[i][j] = max(dp[i-1][j], dp[i][j-1])

✅ Go 实现

func longestCommonSubsequence(text1 string, text2 string) int {
   
    m, n := len(text1), len(text2)
    dp := make([][]int, m+1)
    for i := range dp {
   
        dp[i] = make([]int, n+1)
    }

    for i := 1; i <= m; i++ {
   
        for j := 1; j <= n; j++ {
   
            if text1[i-1] == text2[j-1] {
   
                dp[i][j] = dp[i-1][j-1] + 1
            } else {
   
                dp[i][j] = max(dp[i-1][j], dp[i][j-1])
            }
        }
    }

    return dp[m][n]
}

func max(a, b int) int {
   
    if a > b {
   
        return a
    }
    return b
}

3️⃣ 编辑距离(LeetCode 72)

📌 题目描述

给你两个单词 word1word2,请返回将 word1 转换成 word2 所使用的最少操作数(插入、删除、替换)。


🧠 解题思路:经典 DP 模板题

✅ 状态定义

dp[i][j] 表示将 word1[0...i-1] 转换为 word2[0...j-1] 所需的最小编辑距离。

🔁 状态转移

  • 如果 word1[i-1] == word2[j-1],则 dp[i][j] = dp[i-1][j-1]
  • 否则取三种操作的最小值加 1:

    • 插入:dp[i][j-1] + 1
    • 删除:dp[i-1][j] + 1
    • 替换:dp[i-1][j-1] + 1

✅ Go 实现

func minDistance(word1 string, word2 string) int {
   
    m, n := len(word1), len(word2)
    dp := make([][]int, m+1)
    for i := range dp {
   
        dp[i] = make([]int, n+1)
    }

    for i := 0; i <= m; i++ {
   
        dp[i][0] = i
    }
    for j := 0; j <= n; j++ {
   
        dp[0][j] = j
    }

    for i := 1; i <= m; i++ {
   
        for j := 1; j <= n; j++ {
   
            if word1[i-1] == word2[j-1] {
   
                dp[i][j] = dp[i-1][j-1]
            } else {
   
                dp[i][j] = min(
                    dp[i-1][j-1], // 替换
                    dp[i][j-1],   // 插入
                    dp[i-1][j],   // 删除
                ) + 1
            }
        }
    }

    return dp[m][n]
}

func min(a, b, c int) int {
   
    if a < b {
   
        if a < c {
   
            return a
        }
        return c
    }
    if b < c {
   
        return b
    }
    return c
}

✅ 总结对比

题目 类型 状态定义 状态转移 重点
5 最长回文子串 判断子串是否回文 dp[i][j] 是不是回文 s[i]==s[j] && dp[i+1][j-1] 需倒序遍历
1143 最长公共子序列 序列匹配 dp[i][j] 表示最长长度 相等加一,否则取较大 不能连续字符
72 编辑距离 转换操作 dp[i][j] 表示最小操作数 插入/删除/替换 三选一 是经典题!

目录
相关文章
在markdown中添加视频的两种方法
markdown浏览器中如何添加视频呢?两种方式
|
机器学习/深度学习 JSON 前端开发
RESTful API接口设计规范
近年来移动互联网的发展,前端设备层出不穷(手机、平板、桌面电脑、其他专用设备…),因此,必须有一种统一的机制,方便不同的前端设备与后端进行通信,于是RESTful诞生了,它可以通过一套统一的接口为 Web,iOS和Android提供服务。
4312 1
RESTful API接口设计规范
|
11月前
|
缓存 iOS开发 MacOS
uniapp发布快应用失败报错Error: ENOENT: no such file or directory以及hap-chimera-toolkit问题优雅草卓伊凡
uniapp发布快应用失败报错Error: ENOENT: no such file or directory以及hap-chimera-toolkit问题优雅草卓伊凡
1101 2
uniapp发布快应用失败报错Error: ENOENT: no such file or directory以及hap-chimera-toolkit问题优雅草卓伊凡
|
机器人 Go
【LeetCode 热题100】网格路径类 DP 系列题:不同路径 & 最小路径和(力扣62 / 64 )(Go语言版)
本篇介绍了网格路径类动态规划问题,涵盖 LeetCode 第 62 题“不同路径”和第 64 题“最小路径和”。通过定义状态 `dp[i][j]` 和转移方程,分别解决路径计数与最小代价问题。两题均支持一维数组优化空间复杂度。总结对比了两题的异同,并延伸至带障碍路径、三角形路径等问题,帮助读者掌握网格 DP 的核心思路:明确状态、写出转移、找准边界。
447 48
|
人工智能 自然语言处理 算法
几款宝藏级AI阅读工具推荐!论文分析、文档总结必备神器!
【10月更文挑战第8天】几款宝藏级AI阅读工具推荐!论文分析、文档总结必备神器!
2692 1
几款宝藏级AI阅读工具推荐!论文分析、文档总结必备神器!
|
搜索推荐 前端开发 数据可视化
基于Python协同过滤的旅游景点推荐系统,采用Django框架,MySQL数据存储,Bootstrap前端,echarts可视化实现
本文介绍了一个基于Python协同过滤算法的旅游景点推荐系统,该系统采用Django框架、MySQL数据库、Bootstrap前端和echarts数据可视化技术,旨在为用户提供个性化的旅游推荐服务,提升用户体验和旅游市场增长。
2267 9
基于Python协同过滤的旅游景点推荐系统,采用Django框架,MySQL数据存储,Bootstrap前端,echarts可视化实现
人脸对比
【7月更文挑战第31天】人脸对比
759 2
|
存储 算法 数据挖掘
动态规划Dynamic programming详解-最长公共子序列【python】
动态规划Dynamic programming详解-最长公共子序列【python】
|
存储 算法 安全
HashMap常见面试题(超全面):实现原理、扩容机制、链表何时升级为红黑树、死循环
HashMap常见面试题:红黑树、散列表,HashMap实现原理、扩容机制,HashMap的jd1.7与jdk1.8有什么区别,寻址算法、链表何时升级为红黑树、死循环
|
并行计算 API 异构计算
JAX 中文文档(十六)(2)
JAX 中文文档(十六)
890 1

热门文章

最新文章