每日算法 -【Swift 算法】寻找字符串中最长回文子串(三种经典解法全解析)
🧩 最长回文子串问题:三种经典解法全解析(含代码注释)
本文将系统讲解“最长回文子串”问题的三种常见解法:中心扩展法、动态规划、马拉车算法(Manacher’s Algorithm),并进行对比与总结。
📌 问题描述
给定一个字符串 s
,返回其中 最长的回文子串。
示例:
- 输入:
"babad"
- 输出:
"bab"
或"aba"
✅ 解法一:中心扩展法(Expand Around Center)
🧠 思路
- 回文的本质是“对称”
- 遍历每个字符,尝试以它为中心向两边扩展,总共需要考虑
2n - 1
个中心(包括字符与字符之间的间隙)。 - 需要对奇数长度(如
"aba"
)和偶数长度(如"abba"
)分别处理
⏱ 复杂度
- 时间复杂度:O(n²)
- 空间复杂度:O(1)
🧑💻 Swift 实现(含注释)
func longestPalindromeCenter(_ s: String) -> String {if s.isEmpty { return "" }let chars = Array(s)var start = 0, end = 0for i in 0..<chars.count {let len1 = expand(chars, left: i, right: i) // 奇数长度let len2 = expand(chars, left: i, right: i + 1) // 偶数长度let len = max(len1, len2)if len > end - start {start = i - (len - 1) / 2end = i + len / 2}}return String(chars[start...end])
}func expand(_ chars: [Character], left: Int, right: Int) -> Int {var L = left, R = rightwhile L >= 0 && R < chars.count && chars[L] == chars[R] {L -= 1R += 1}return R - L - 1 // 实际回文长度
}
✅ 解法二:动态规划(Dynamic Programming)
🧠 思路
- 定义
dp[i][j]
表示s[i...j]
是否为回文串。 - 转移方程:
dp[i][j] = true
ifs[i] == s[j] && dp[i+1][j-1]
⏱ 复杂度
- 时间复杂度:O(n²)
- 空间复杂度:O(n²)
🧑💻 Swift 实现(含注释)
func longestPalindromeDP(_ s: String) -> String {let chars = Array(s)let n = chars.countif n < 2 { return s }var dp = Array(repeating: Array(repeating: false, count: n), count: n)var maxLen = 1var start = 0for i in 0..<n {dp[i][i] = true}for length in 2...n {for i in 0...(n - length) {let j = i + length - 1if chars[i] == chars[j] {if length == 2 {dp[i][j] = true} else {dp[i][j] = dp[i + 1][j - 1]}if dp[i][j] && length > maxLen {maxLen = lengthstart = i}}}}return String(chars[start..<start + maxLen])
}
✅ 解法三:马拉车算法(Manacher’s Algorithm)
🧠 思路
- 首先对字符串预处理,使得所有回文都是奇数长度(例如:
"abba"
→"#a#b#b#a#"
) - 使用数组
p[i]
记录以第i
个字符为中心的最大回文“半径” - 利用已知的对称中心和边界进行跳跃扩展,从而实现线性时间复杂度
⏱ 复杂度
- 时间复杂度:O(n)
- 空间复杂度:O(n)
🧑💻 Swift 实现(含注释)
func longestPalindromeManacher(_ s: String) -> String {// Step 1: 预处理字符串,在每个字符中间加入 #let t = Array("#" + s.map { "\($0)#" }.joined())let n = t.countvar p = Array(repeating: 0, count: n)var center = 0, right = 0var maxLen = 0, maxCenter = 0for i in 0..<n {let mirror = 2 * center - i // i 关于 center 的对称点// Step 2: 利用镜像对称性快速扩展if i < right {p[i] = min(right - i, p[mirror])}// Step 3: 尝试向左右扩展var a = i + p[i] + 1var b = i - p[i] - 1while a < n && b >= 0 && t[a] == t[b] {p[i] += 1a += 1b -= 1}// Step 4: 更新 center 和 rightif i + p[i] > right {center = iright = i + p[i]}// Step 5: 更新最大值if p[i] > maxLen {maxLen = p[i]maxCenter = i}}// Step 6: 从原始字符串中提取结果let start = (maxCenter - maxLen) / 2let end = start + maxLenreturn String(s[s.index(s.startIndex, offsetBy: start)..<s.index(s.startIndex, offsetBy: end)])
}
🧪 示例输出
print(longestPalindromeCenter("babad")) // 输出: "bab" 或 "aba"
print(longestPalindromeManacher("cbbd")) // 输出: "bb"
print(longestPalindromeDP("babad")) // "bab" 或 "aba"
📊 三种方法对比总结
算法 | 时间复杂度 | 空间复杂度 | 实现难度 | 适用场景 |
---|---|---|---|---|
中心扩展法 | O(n²) | O(1) | ⭐⭐ 易 | 面试首选,简洁实用 |
动态规划 | O(n²) | O(n²) | ⭐⭐⭐ 中等 | 适用于变种题型 |
马拉车算法 | O(n) | O(n) | ⭐⭐⭐⭐ 高 | 性能关键、竞赛场景 |
✅ 最佳实践推荐
需求 | 推荐方法 |
---|---|
面试、日常开发、可读性 | ✅ 中心扩展法 |
遇到类似子串变种问题 | ✅ 动态规划 |
超大数据量、刷算法题、竞赛 | ✅ 马拉车算法 |
相关文章:
每日算法 -【Swift 算法】寻找字符串中最长回文子串(三种经典解法全解析)
🧩 最长回文子串问题:三种经典解法全解析(含代码注释) 本文将系统讲解“最长回文子串”问题的三种常见解法:中心扩展法、动态规划、马拉车算法(Manacher’s Algorithm),并进行对比与…...
《Cesium全生态解析:从入门到精通的3D地理空间开发指南》
在WebGL、GIS和三维可视化技术高速发展的今天,Cesium 已经从一个开源地图引擎成长为全球开发者构建数字地球的核心工具。从地球到火星,从网页到游戏引擎,Cesium以其跨平台、高精度和无限扩展性,重新定义了我们对空间数据的交互方式…...
pytorch LSTM 结构详解
最近项目用到了LSTM ,但是对LSTM 的输入输出不是很理解,对此,我详细查找了lstm 的资料 import torch.nn as nnclass LSTMModel(nn.Module):def __init__(self, input_size1, hidden_size50, num_layers2):super(LSTMModel, self).__init__()…...

流程自动化引擎:重塑企业数字神经回路
在数字经济高速发展的今天,企业运营的核心逻辑正在经历一场静默的革命。流程自动化引擎作为这场变革的中枢神经系统,通过智能化的技术手段重构企业的业务逻辑与决策链路,将原本离散的“数字神经元”编织成高效协同的神经网络。这种技术不仅打…...

nginx web服务日志分析
特点: 实时分析:支持实时分析 Nginx 日志,无需预先存储大量日志数据,能即时反馈网站的访问情况。轻量级高效:资源占用少,运行速度快,适合处理高流量网站的日志分析。多种输出格式:除…...

VSCode+EIDE通过KeilC51编译,使VSCode+EIDE“支持”C和ASM混编
在使用Keil C51时,要让Keil C51支持混编则需要在混编的.c文件上右键选择Options for File *(ALTF7),打开选项界面后,在 Properties 页 勾上 Generate Assembler SRC File 和 Assemble SRC File ,如下图所示: 这样设置后…...
5.23本日总结
一、英语 复习list5list25 二、数学 写14讲部分课后题,学习15讲部分 三、408 写计网5.3题目,学习计组第一章 四、总结 二重积分的题目也涉及了一元函数积分相关知识,该部分遗忘较多,后续需要再复习。 五、明日计划 英语&…...

游戏引擎学习第298天:改进排序键 - 第1部分
关于向玩家展示多个房间层所需的两种 Z 值 我们在前一天基本完成了为渲染系统引入分层 Z 值的工作,但还没有完全完成所有细节。我们开始引入图形渲染中的分层概念,即在 Z 轴方向上拥有多个独立图层,每个图层内部再使用一个单独的 Z 值来实现…...

Mysql篇-优化
Mysql篇主要是纯理论的面试问题与技巧。 主要从以下进行开展: 索引相关问题: 1、Mysql如何定位慢查询? Mysql慢查询:某个业务查询数据响应时间过长或者与预期响应时间相差大。 表象:页面加载过慢、接口压测响应时间…...
Java 集合框架核心知识点全解析:从入门到高频面试题(含 JDK 源码剖析)
一、Java 集合框架体系架构 Java 集合框架分为两大分支: Collection接口:存储单个元素,包括: List:有序、可重复(如ArrayList、LinkedList)Set:无序、唯一(如HashSet、…...

一文详解生成式 AI:李宏毅《生成式 AI 导论》学习笔记
生成式 AI 是怎么回事 人工智能(Artificial Intelligence) “智能”是一个广泛而复杂的概念,其定义和应用范围随着技术、科学和社会的发展不断演变。在当前的语境下,“智能”通常与人工智能(AI)相关联&am…...

什么是物联网 (IoT):2024 年物联网概述
物联网(IoT)是一个有望彻底改变我们生活、工作以及与环境互动方式的概念。如今,越来越多的新兴企业和老牌企业都在利用物联网的力量创造创新产品与服务。正因为这一转变,互联互通已成为我们生活中不可或缺的一部分,科技…...

8级-数组
前情回顾:在7级的时候,我们学习了如何定义、使用函数 目录 概念 什么是数组? 一维数组 声明 初始化 访问元素 计算数组长度 二维数组 声明 初始化 访问元素 思考 一维数组在内存中如何存储? 二维数组在内存中如何存储&…...

大模型 Agent 就是文字艺术吗?
最近在技术圈里有一个很有趣的争论:大模型 Agent 是不是就是各种 Prompt 的堆叠?像 Manus 这样看起来很智能的 Agent,本质上是不是就是用巧妙的 Prompt 约束大模型生成更好的输出?换句话说,这是不是一门文字艺术&#…...

YOLOv8检测头代码详解(示例展示数据变换过程)
本文旨在通过实例数据,详细解读YOLOv8检测头的网络结构及其代码实现。首先将从检测头的网络架构开始讲解,涵盖代码与网络结构图的对比分析。关键在于深入探讨检测头的输出结果,因为这些输出将直接用于损失函数的计算。由于在不同阶段…...

JUC并发编程1
什么是juc 在java的java.util.concurrent包下的工具。 锁 传统的synchronize public class SealTicket {int total 50;public synchronized void seal() {if (total > 0) {System.out.println(Thread.currentThread().getName() "卖出第" (total--) "张…...
消息队列RabbitMQ与AMQP协议详解
消息队列RabbitMQ与AMQP协议详解 什么是RabbitMQ RabbitMQ是一个开源的消息队列中间件,基于AMQP(Advanced Message Queuing Protocol)协议实现。它作为一个消息代理(Message Broker),可以接收、存储和转发…...
Day 29 训练
Day 29 训练 Day 29:Python 类装饰器的奥秘与实践一、类装饰器:函数装饰器的升级版二、类装饰器 VS 函数装饰器:核心区别三、实战:为类添加日志功能四、类方法定义的两种风格1. 类内部定义方法(常规方式)2.…...

STM32开发环境配置——VSCode+PlatformIO + CubeMX + FreeRTOS的集成环境配置
前言 为什么配置这样的一个环境呢?鄙人受够了Keil5那个简陋的工作环境了,实在是用不下去,调试上很容易跟CubeMX的代码产生不协调导致调试——发布代码不一致造成的一系列问题。CubeIDE虽说不错,但是它的代码辅助功能和构建系统实在…...

Profibus转Profinet网关赋能鼓式硫化机:智能化生产升级的关键突破
在现代工业自动化领域,通讯协议转换器发挥着至关重要的角色。它们能够实现不同网络间的无缝对接和数据传输,确保了生产线上的设备可以顺畅地交流信息。今天,我们就来深入讨论开疆智能profibus转profinet网关KJ-PBM-PN以及其在鼓式硫化机中的应…...
redis 缓存穿透,缓存雪崩,缓存击穿
之前也不知道是哪个老六总结出来得缓存穿透,缓存击穿 。 穿透,击穿 中文上容易搞混,所以贴出英文 缓存穿透: Cache Penetration “Penetration” 有穿透、渗透之意, eg: the penetration of hackers into the system (黑客对系统的侵入) 缓…...
JAVA8怎么使用9的List.of
在 Java 8 中,List.of 方法并不可用,因为这是从 Java 9 开始引入的用于创建不可变列表的便捷方法。要在 Java 8 中达到类似的效果,您需要使用其他方式来创建列表。常规的方法是先创建集合对象然后再添加元素 List<String> list new A…...
告别手动测试:AUTOSAR网络管理自动化测试实战
文章目录 一、自动化测试系统架构硬件组成软件架构 二、测试覆盖的关键场景状态机测试时间参数测试容错性测试 三、测试case举例四、小结 一、自动化测试系统架构 AUTOSAR网络管理自动化测试由硬件设备和软件工具共同完成。 硬件组成 程控电源(DUT供电࿰…...
BUCK电路利用状态空间平均法和开关周期平均法推导
以BUCK电路为例的两种方法推导 BUCK电路简介 BUCK电路是一种降压型DC-DC转换器,其拓扑结构如下: 输入电压 V in V_{\text{in}} Vin,输出电压 V out = D V in V_{\text{out}} = D V_{\text{in}} Vout=DVin(稳态时, D D D为占空比)。关键元件:开关管 S S S、续流…...

MongoDB 用户与权限管理完全指南
在当今数据驱动的时代,数据库安全已成为企业IT架构中最关键的环节之一。作为最受欢迎的NoSQL数据库之一,MongoDB提供了完善的用户认证和权限管理机制,但许多开发者和数据库管理员对这些功能的理解和应用仍停留在表面层次。本文将全面剖析Mong…...
C++滑动门问题(附两种方法)
题目如下: 滑动窗口 - 题目 - Liusers OJ ——引用自OJ网站 方法如下: 1.常规思想 #include<bits/stdc.h> using namespace std; int main() {int n,k;int a[110];cin>>n>>k;for(int i0;i<n;i){cin>>a[i];}for(int i0;i…...

基于ITcpServer/IHttpServer框架的HTTP服务器
https://www.cnblogs.com/MuZhangyong/p/16839231.html 在基于ITcpServer/IHttpServer框架的HTTP服务器实现中,OnBody方法主要用于接收HTTP请求体数据,而触发HTTP响应通常是在OnMessageComplete方法中完成。以下是完整的响应触发机制说明: sequenceDiagramClient->>…...
初识main函数
int main(int argc, char *argv[]) {int a 0;return a; }X64 MSVC编译器 Windows x64调用约定 { // 将第二个参数(rdx)保存到栈[rsp0x10]位置 0x7ff6e54c2ad0 mov qword ptr [rsp10h],rdx // 将第一个参数(ecx)保存到栈[rsp8]位置 0x7ff6e54c2ad5 …...

FPGA高效验证工具Solidify 8.0:全面重构图形用户界面
近日,FPGA高效验证工具Solidify发布了8.0版本。该版本对图形用户界面(GUI)进行了全面重构,历时两年,经过了大幅的架构改进,旨在为用户提供更安全、更稳定的使用环境。 Solidify的用户对隐私有严格要求&…...

SIL2/PLd 认证 Inxpect毫米波安全雷达:3D 扫描 + 微小运动检测守护工业安全
Inxpect 成立于意大利,专注工业安全技术。自成立起,便致力于借助先进雷达技术提升工业自动化安全标准,解决传统安全设备在复杂环境中的局限,推出获 SIL2/PLd 和 UL 认证的安全雷达产品。 Inxpect 的雷达传感器技术优势明显。相较于…...