当前位置: 首页 > 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.lryc.cn/news/609577.html

相关文章:

  • 【YOLO系列】YOLOv12详解:模型结构、损失函数、训练方法及代码实现
  • 关于Npm和Nvm的用法
  • Linux 环境 libpq加载异常导致psql 连接 PostgreSQL 库失败失败案例
  • uniapp开发微信小程序textarea在ios下有默认内边距的问题(textarea兼容问题)
  • 如何给Word和WPS文档添加密码或取消密码
  • Ethereum:拥抱开源,OpenZeppelin 未来的两大基石 Relayers 与 Monitor
  • Jwts用于创建和验证 ​​JSON Web Token(JWT)​​ 的开源库详解
  • OpenLayers 入门指南【五】:Map 容器
  • R 语言科研绘图第 67 期 --- 箱线图-显著性
  • Nestjs框架: Node.js 多环境配置策略与 dotenv 与 config 库详解
  • 政府财政行业云原生转型之路
  • Druid学习笔记 01、快速了解Druid中SqlParser实现
  • 排序算法入门:直接插入排序详解
  • 室内分布系统
  • ICCV 2025|单视频生成动态4D场景!中科大微软突破4D生成瓶颈,动画效果炸裂来袭!
  • Flutter开发 了解Scaffold
  • 深入理解Java的SPI机制,使用auto-service库优化SPI
  • 区块链基础之Merkle B+树
  • Azure DevOps - 使用 Ansible 轻松配置 Azure DevOps 代理 - 第6部分
  • 打造个人数字图书馆:LeaNote+cpolar如何成为你的私有化知识中枢?
  • 多级表头的导出
  • 软件打包前进行文件去重
  • Unix 命令行shell基础--学习系列003
  • Web 开发 12
  • 嵌入式硬件中三极管原理分析与控制详解
  • 嵌入式硬件篇---OpenMV存储
  • 单片机51 day46
  • 基于单片机智能鱼缸/水族箱/水产养殖系统设计
  • 第二篇:深入解析 FastAPI + LangChain 实现流式对话接口:`chat` 函数详解
  • 嵌入式硬件中三极管推挽电路控制与实现