当前位置: 首页 > news >正文

LeetCode 132:分割回文串 II

LeetCode 132:分割回文串 II

在这里插入图片描述

问题本质与核心挑战

给定字符串 s,需将其分割为若干回文子串,求最少分割次数。核心挑战:

  • 直接枚举所有分割方式(指数级复杂度)不可行;
  • 需结合 动态规划 优化分割次数计算,并通过 回文预处理 加速判断。

核心思路:动态规划 + 回文预处理

1. 回文预处理(减少重复判断)

用二维数组 isPal[i][j] 记录 子串 s[i..j] 是否为回文,预处理后可 O(1) 查询:

  • 单个字符isPal[i][i] = true
  • 两个字符isPal[i][i+1] = (s[i] == s[i+1])
  • 长度 ≥3isPal[i][j] = (s[i] == s[j] && isPal[i+1][j-1])(依赖更短的子串结果)。
2. 动态规划定义与转移
  • 状态定义dp[i] 表示 i 个字符(s[0..i-1] 的最少分割次数。
  • 初始条件
    • dp[0] = -1(空字符串的分割次数为 -1,方便后续计算);
    • dp[i] = i-1(最坏情况:每个字符单独分割,如 "abc" 需要 2 次分割)。
  • 状态转移
    对于每个 j(前 j 个字符),遍历所有可能的分割点 i0 ≤ i < j):
    • s[i..j-1] 是回文(isPal[i][j-1] = true),则 dp[j] = min(dp[j], dp[i] + 1)

算法步骤详解(以示例 s = "aab" 为例)

步骤 1:预处理回文子串(isPal 数组)
子串范围 [i,j]长度判断逻辑isPal[i][j]
[0,0]1单个字符true
[1,1]1单个字符true
[2,2]1单个字符true
[0,1]2s[0]='a' == s[1]='a'true
[1,2]2s[1]='a' ≠ s[2]='b'false
[0,2]3s[0]≠s[2](直接不满足)false
步骤 2:初始化动态规划数组(dp
  • dp[0] = -1(空字符串的分割次数);
  • dp[1] = 0(前1个字符 "a",无需分割);
  • dp[2] = 1(初始值,后续会被更新);
  • dp[3] = 2(初始值,后续会被更新)。
步骤 3:状态转移计算

遍历 j(前 j 个字符)和 i(分割点):

j(前j字符)i(分割点)子串 s[i..j-1]是否回文(isPal[i][j-1]dp[i] + 1dp[j] 更新后的值
j=1i=0"a"true-1 + 1 = 0min(0, 0) = 0
j=2i=0"aa"true-1 + 1 = 0min(1, 0) = 0
j=2i=1"a"true0 + 1 = 1min(0, 1) = 0
j=3i=0"aab"false-不更新
j=3i=1"ab"false-不更新
j=3i=2"b"true0 + 1 = 1min(2, 1) = 1

完整代码(Java)

class Solution {public int minCut(String s) {int n = s.length();if (n == 0) return 0;// 步骤1:预处理回文子串boolean[][] isPal = new boolean[n][n];// 处理长度为1的回文for (int i = 0; i < n; i++) {isPal[i][i] = true;}// 处理长度为2的回文for (int i = 0; i < n - 1; i++) {isPal[i][i + 1] = (s.charAt(i) == s.charAt(i + 1));}// 处理长度≥3的回文for (int len = 3; len <= n; len++) {for (int i = 0; i + len <= n; i++) {int j = i + len - 1;isPal[i][j] = (s.charAt(i) == s.charAt(j) && isPal[i + 1][j - 1]);}}// 步骤2:动态规划int[] dp = new int[n + 1];// 初始条件:最坏情况,每个字符单独分割for (int i = 0; i <= n; i++) {dp[i] = i - 1;}// 状态转移:遍历前j个字符,尝试所有分割点ifor (int j = 1; j <= n; j++) {for (int i = 0; i < j; i++) {if (isPal[i][j - 1]) { // s[i..j-1]是回文dp[j] = Math.min(dp[j], dp[i] + 1);}}}return dp[n];}
}

关键逻辑解析

  1. 回文预处理:通过动态规划预处理所有子串的回文性,避免每次判断回文时重复计算,时间复杂度 O(n²)
  2. 动态规划状态dp[j] 表示前 j 个字符的最少分割次数,利用已计算的 dp[i] 快速推导,时间复杂度 O(n²)
  3. 初始条件优化dp[0] = -1 是为了让 dp[1] 的计算更自然(dp[0] + 1 = 0,对应单个字符无需分割)。

该方法通过 预处理+动态规划 高效解决问题,时间复杂度为 O(n²),可处理题目中 n ≤ 2000 的规模。核心是将“回文判断”和“分割次数计算”解耦,通过预处理降低重复判断的开销,再利用动态规划的状态转移快速推导最优解。

http://www.xdnf.cn/news/1243207.html

相关文章:

  • Linux开发利器:探秘开源,构建高效——基础开发工具指南(下)【make/Makefile】
  • 水面垃圾清扫船cad【6张】三维图+设计说明书
  • Jmeter进行性能并发测试
  • 【Java】使用FreeMarker来实现Word自定义导出
  • C++高频知识点(十四)
  • 京东商品详情API技术文档框架及Python实现方案
  • sqli-labs:Less-27a关卡详细解析
  • 《Python 实用项目与工具制作指南》· 2.3 导入
  • Bean的生命周期和循环依赖问题的解决
  • curl发送文件bodyParser无法获取请求体的问题分析
  • 嵌入式硬件中三极管推挽电路控制与实现
  • PPT自动化 python-pptx - 11 : 备注页 (Notes Slides)
  • (论文速读)Text-IF:基于语义文本引导的退化感知交互式图像融合方法
  • sqli-labs-master/Less-31~Less-40
  • openeuler离线安装软件
  • Hexo - 免费搭建个人博客07 - 添加右上角的“目录”
  • 先知模型或者说从容的模型
  • Linux—yum仓库及NFS网络共享服务
  • Java基础-斗地主游戏
  • opencv引入libavif
  • 从 0 到 1 开发图书管理系统:飞算 JavaAI 让技术落地更简单
  • Prometheus-3--Prometheus是怎么抓取Java应用,Redis中间件,服务器环境的指标的?
  • 【慕伏白】Android Studio 配置国内镜像源
  • 内联函数:提升效率的空间换时间艺术
  • FreeRTOS源码分析四:时钟中断处理响应流程
  • 深入浅出 RabbitMQ:工作队列实战(轮训策略VS公平策略)
  • 鸿蒙南向开发 编写一个简单子系统
  • 机器学习 入门——决策树分类
  • 并发编程常用工具类(下):CyclicBarrier 与 Phaser 的协同应用
  • C++入门自学Day6-- C++模版