面试算法题精讲:最长回文子串

news2025/1/12 6:00:33

面试算法题精讲:最长回文子串

题目来源:5. 最长回文子串

题目描述:

给你一个字符串 s,找到 s 中最长的回文子串。

如果字符串的反序与原始字符串相同,则该字符串称为回文字符串。

解法1:动态规划

对于一个子串而言,如果它是回文串,并且长度大于 2,那么将它首尾的两个字母去除之后,它仍然是个回文串。

根据这样的思路,我们就可以用动态规划的方法解决本题。

我们用 dp[i][j] 表示 s[i…j] 是否是回文串。

状态转移方程:dp[i][j] = dp[i+1][j-1] ∧ (s[i] == s[j])

动态规划中的边界条件:

  1. dp[i][i] == true,对于长度为 1 的子串,它显然是个回文串。
  2. dp[i][i+1] = (s[i] == s[i+1]),对于长度为 2 的子串,只要它的两个字母相同,它就是一个回文串。

根据这个思路,我们就可以完成动态规划了,最终的答案即为所有 dp[i][j]=true 中 j−i+1(即子串长度)的最大值。注意:在状态转移方程中,我们是从长度较短的字符串向长度较长的字符串进行转移的,因此一定要注意动态规划的循环顺序。

我们使用2层循环,外层枚举子串的长度 len,内层枚举子串起点 i,子串的终点 j = i + len - 1。之后进行转移转移即可。

代码:

class Solution
{
public:
    string longestPalindrome(string s)
    {
        // 特判
        if (s.size() < 2)
            return s;
        int n = s.size(), maxLen = 1, begin = 0;
        // 状态矩阵
        vector<vector<int>> dp(n, vector<int>(n, false));
        // dp[i][j] 表示 s[i...j] 是否是回文串
        // 初始化:所有长度为 1 的子串都是回文串
        for (int i = 0; i < n; i++)
            dp[i][i] = true;
        // 状态转移
        for (int len = 2; len <= n; len++) // 枚举子串长度
            for (int i = 0; i < n; i++)    // 枚举左边界
            {
                int j = i + len - 1; // 计算右边界
                if (j >= n)          // 右边界越界
                    break;
                if (s[i] != s[j])
                    dp[i][j] = false;
                else
                {
                    if (len <= 3)
                        dp[i][j] = true;
                    else
                        dp[i][j] = dp[i + 1][j - 1];
                }
                if (dp[i][j] == true && j - i + 1 > maxLen)
                {
                    maxLen = j - i + 1;
                    begin = i;
                }
            }
        return s.substr(begin, maxLen);
    }
};

结果:

在这里插入图片描述

复杂度分析:

时间复杂度:O(n2),其中 n 是字符串 s 的长度。

空间复杂度:O(n2),其中 n 是字符串 s 的长度。

解法2:中心拓展算法

「中心扩散法」的基本思想是:遍历每一个下标,以这个下标为中心,利用「回文串」中心对称的特点,往两边扩散,看最多能扩散多远。

从每一个位置出发,向两边扩散即可。遇到不是回文的时候结束。

每个位置向两边扩散都会出现一个窗口大小(len = right - left)。如果 len>maxLen(用来表示最长回文串的长度),则更新 maxLen 的值。

因为我们最后要返回的是具体子串,而不是长度。因此,还需要记录一下 maxLen 时的起始位置 start。

代码:

/*
 * @lc app=leetcode.cn id=5 lang=cpp
 *
 * [5] 最长回文子串
 */

// @lc code=start
class Solution
{
public:
    string longestPalindrome(string s)
    {
        int n = s.length();
        int start = 0, end = 0;

        auto expendAroundCenter = [&](int left, int right) -> pair<int, int>
        {
            while (left >= 0 && right < n && s[left] == s[right])
            {
                left--;
                right++;
            }
            return {left + 1, right - 1};
        };

        for (int i = 0; i < n; i++)
        {
            auto [left1, right1] = expendAroundCenter(i, i);
            if (right1 - left1 > end - start)
            {
                start = left1;
                end = right1;
            }
            auto [left2, right2] = expendAroundCenter(i, i + 1);
            if (right2 - left2 > end - start)
            {
                start = left2;
                end = right2;
            }
        }
        int len = end - start + 1;
        return s.substr(start, len);
    }
};
// @lc code=end

结果:

在这里插入图片描述

复杂度分析:

时间复杂度:O(n2),其中 n 是字符串 s 的长度。

空间复杂度:O(1)。

解法3:Manacher 算法

算法详解:https://blog.csdn.net/dyx404514/article/details/42061017。

代码:

class Solution
{
private:
    string manacher(string s)
    {
        // 特判
        if (s.empty() || s.size() < 2)
            return s;
        // 对原始字符串 s 做处理,添加分隔符(例如:将 abc 变成 #a#b#c#)
        string str = addBoundaries(s, '#');
        int n = str.size();
        // right 表示已经探测到的字符串最右边的可达范围
        int right = 0;
        // center 表示根据最右边的可达范围的中心对称位置
        int center = 0;
        int start = 0, maxLen = 0;
        // p 数组记录所有已探测过的回文半径,后面我们再计算 i 时,根据 p[i_mirror] 计算 i
        vector<int> p(n, 0);
        // 从左到右遍历处理过的字符串,求每个字符的回文半径
        for (int i = 0; i < n; i++)
        {
            // 根据i和right的位置分为两种情况:
            // 1. i <= right,利用已知的信息来计算 i
            // 2. i > right,说明 i 的位置时未探测过的,只能用中心探测法
            if (right >= i)
            {
                // 这句是关键,不用再像中心探测那样,一点点的往左/右扩散,根据已知信息
                // 减少不必要的探测,必须选择两者中的较小者作为左右探测起点
                int minArmLen = min(right - i, p[2 * center - i]);
                p[i] = expand(str, i - minArmLen, i + minArmLen);
            }
            else // i 落在 right 右边,是没被探测过的,只能用中心探测法
                p[i] = expand(str, i, i);
            // 大于right,说明可以更新最右端范围了,同时更新 center
            if (i + p[i] > right)
            {
                center = i;
                right = i + p[i];
            }
            // 找到了一个更长的回文半径,更新原始字符串的 start 位置
            if (p[i] > maxLen)
            {
                maxLen = p[i];
                start = (i - p[i]) / 2;
            }
        }
        // 根据 start 和 maxLen ,从原始字符串中截取一段返回
        return s.substr(start, maxLen);
    }
    // 辅函数 - 以s [left...right] 为起点,计算回文半径(可拓展的步数)
    int expand(string s, int left, int right)
    {
        while (left >= 0 && right < s.size() && s[left] == s[right])
        {
            left--;
            right++;
        }
        // 由于while循环退出后left和right各多走了一步,所以在返回的总长度时要减去2
        return (right - left - 2) / 2;
    }
    // 辅函数 - 对原始字符串 s 进行预处理(添加分隔符)
    string addBoundaries(string s, char divide)
    {
        if (s.empty())
            return "";
        string t;
        for (char &c : s)
        {
            t += divide;
            t += c;
        }
        t += divide;
        return t;
    }

public:
    string longestPalindrome(string s)
    {
        return manacher(s);
    }
};

结果:

在这里插入图片描述

复杂度分析:

时间复杂度:O(n),其中 n 是字符串 s 的长度。

空间复杂度:O(n),其中 n 是字符串 s 的长度。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1629256.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

ubuntu22.04安装TensorRT(过程记录)

重要说明&#xff1a;此贴经过多次修改。第一次安装的的为trt8.6.1版本。第二次安装的10.0.0.6版本。有些地方可能没改过来&#xff0c;比如链接向导&#xff0c;我懒得改了&#xff0c;但是流程是对的。 cuda和cudnn版本对应关系 tensorRT历史发行版本 CUDA历史发行版本 cudn…

【Godot4.2】有序和无序列表函数库 - myList

概述 在打印输出或其他地方可能需要构建有序或无序列表。本质就是构造和维护一个纯文本数组。并用格式化文本形式&#xff0c;输出带序号或前缀字符的多行文本。 为此我专门设计了一个类myList&#xff0c;来完成这项任务。 代码 以下是myList类的完整代码&#xff1a; # …

Android 设置头像 - 相册拍照

Android开发在个人信息管理中&#xff0c;如果设置头像&#xff0c;一般都提供了从相册选择和拍照两种方式。下午将针对设置用户头像相册和拍照两种方式的具体实现进行详细说明。 在实际实现过程中需要使用到权限管理&#xff0c;新版本的Android需要动态申请权限&#xff0c;权…

【JAVA】一文掌握Java并发编程

Java 开发中&#xff0c;并发编程属于相当重要的一个知识点&#xff0c;可以说&#xff0c;Java 的并发能力&#xff0c;是成就今日 Java 地位的因素之一。Java 的并发编程由浅入深实质上是包含 Java&#xff08;API&#xff09;层、JVM&#xff08;虚拟机&#xff09;层、内核…

Mac下使用homebrew管理多版本mysql同时启动

Mac下使用homebrew管理多版本mysql同时启动 思路 给每个版本分配不同的数据目录和配置文件即可 本文尝试了使用 brew 安装管理多个MySQL版本&#xff0c;同时运行、直接切换 安装 如果已有数据文件请自行备份以及使用 安装 mysql 5.7 brew install mysql5.7在 /opt/home…

运维笔记:基于阿里云跨地域服务器通信(上)

运维笔记 阿里云&#xff1a;跨地域服务器通信&#xff08;上&#xff09; - 文章信息 - Author: 李俊才 (jcLee95) Visit me at CSDN: https://jclee95.blog.csdn.netMy WebSite&#xff1a;http://thispage.tech/Email: 291148484163.com. Shenzhen ChinaAddress of this a…

C语言编译的优化等级应该选哪个?O0、O1、O2还是O3

在使用IDE开发STM32程序时&#xff0c;IDE一般都会提供优化等级设置的选项&#xff0c;例如下图中KEIL软件优化等级的设置。 从上图中也可以看出&#xff0c;设置不同的优化等级&#xff0c;实际上是修改了编译器的编译参数。这个编译器是由ARM公司提供的C/C编译器armclang或者…

opencv4.8 系列一环境搭搭建

open 运行环境&#xff1a; vs2017 下载地址&#xff1a;https://www.123pan.com/s/cVyRVv-ydPWh.html 一&#xff1a;新建项目 二&#xff1a;核心代码&#xff1a; 在这里插入代码片 #include<opencv2/opencv.hpp>int main(int argc,char** argv) {cv::Mat src cv…

【软考高项】二十六、范围管理基础内容

一、管理基础 产品范围和项目范围 产品范围强调结果&#xff0c;项目范围强调结果 管理的新实践 &#xff1a;需求一直是项目管理的关注重点&#xff0c;需求管理过程结束于需求关闭&#xff0c;即把产品、服务或成果移交给接收方&#xff0c;以便长期测量、监控、实现并维持收…

ptyhon画图显示中文

import matplotlib.pyplot as plt import matplotlib# 设置中文字体 matplotlib.rcParams[font.sans-serif] [SimHei] matplotlib.rcParams[font.family]sans-serifplt.plot([1, 2, 3, 4]) plt.xlabel(这是x轴) plt.ylabel(这是y轴) plt.title(这是标题) plt.show()用这个代码…

anaconda安装python 3.8环境

打开anaconda命令行窗口 在命令行窗口中&#xff0c;输入命令&#xff1a;conda create -n py38 python3.8 执行命令后&#xff0c;显示conda版本、安装路径和安装的包 然后提醒是否安装&#xff0c;输入y 等待安装完成。然后进入python3.8&#xff0c;执行命令&#xff1a;con…

收藏:什么是协程的通俗解析

不错的视频&#xff1a;到底该怎么理解协程&#xff1f;_哔哩哔哩_bilibili 重点的要点&#xff1a; 比如这个函数&#xff1a; python中&#xff0c;使用yield关键字来做协程&#xff0c;就是暂停可以去执行其他东西&#xff0c;然后其他东西执行完后&#xff0c;继续执行yiel…

抓包理解协议

用的Wireshark 抓包 1.抓包网卡选择 - WLAN 无线网卡&#xff0c;其他是本地虚拟机的网卡 这里分别是开始捕获、停止捕获、重新捕获、网卡选择&#xff0c;下面是可以过滤选择 过滤tcp包 3次握手&#xff1a; source是源地址&#xff0c; destination是目标地址&#xff0c;in…

Mysql用语句创建表/插入列【示例】

一、 创建表 COMMENT表示字段或列的注释 -- 新建student表 CREATE TABLE student (id BIGINT NOT NULL COMMENT 学生id, enroll_date DATE NOT NULL COMMENT 注册时间, NAME VARCHAR(18) DEFAULT NOT NULL COMMENT 学生姓名, deal_flag TINYINT(1) DEFAULT 0 NOT NULL COMM…

创新入门|从点击到转化:AI个性化登陆页助力潜在客户转化

在数字营销的竞争格局中&#xff0c;采用先进技术对于旨在区分自己并吸引受众的企业至关重要。人工智能 &#xff08;AI&#xff09; 成为一项关键技术&#xff0c;尤其是在制作个性化登录页面的艺术方面。这些页面不仅仅是品牌与其潜在客户之间的第一个接触点;它们是吸引兴趣、…

vue-admin-template项目实现中英文切换

实现效果&#xff1a; 1.安装 *注意版本号 npm install vue-i18n8.24.5 -S2.新建文件夹 在src目录下新建lang文件夹&#xff0c;里面有3个文件 // index.js import Vue from vue import VueI18n from vue-i18n import Cookies from js-cookie import elementEnLocale fr…

Redis 服务等过期策略和内存淘汰策略解析

redis服务是基于内存运行的&#xff0c;所以很多数据都存放在内存中&#xff0c;但是内存又不是无限的&#xff0c;所以redis就引出了key的过期和淘汰策略。 一、Redis的过期策略&#xff1a; 我们在set key的时候&#xff0c;可以给它设置一个过期时间&#xff0c;比如expire …

【UnityRPG游戏制作】RPG项目的背包系统商城系统和BOSS大界面

&#x1f468;‍&#x1f4bb;个人主页&#xff1a;元宇宙-秩沅 &#x1f468;‍&#x1f4bb; hallo 欢迎 点赞&#x1f44d; 收藏⭐ 留言&#x1f4dd; 加关注✅! &#x1f468;‍&#x1f4bb; 本文由 秩沅 原创 &#x1f468;‍&#x1f4bb; 收录于专栏&#xff1a;Uni…

【06】JAVASE-数组讲解【从零开始学JAVA】

Java零基础系列课程-JavaSE基础篇 Lecture&#xff1a;波哥 Java 是第一大编程语言和开发平台。它有助于企业降低成本、缩短开发周期、推动创新以及改善应用服务。如今全球有数百万开发人员运行着超过 51 亿个 Java 虚拟机&#xff0c;Java 仍是企业和开发人员的首选开发平台。…

华为MRS服务使用记录

背景&#xff1a;公司的业务需求是使用华为的这一套成品来进行开发&#xff0c;使用中发现&#xff0c;这个产品跟原生的Hadoop的那一套的使用&#xff0c;还是有很大的区别的&#xff0c;现记录一下&#xff0c;避免以后忘了 一、原始代码的下载 下载地址&#xff1a;MRS样例…