程序员面试题精选100题(46)-对称子字符串的最大长度

2023-05-16

程序员面试题精选100题(46)-对称子字符串的最大长度

题目:输入一个字符串,输出该字符串中对称的子字符串的最大长度。比如输入字符串“google”,由于该字符串里最长的对称子字符串是“goog”,因此输出4。

分析:可能很多人都写过判断一个字符串是不是对称的函数,这个题目可以看成是该函数的加强版。

引子:判断字符串是否对称

要判断一个字符串是不是对称的,不是一件很难的事情。我们可以先得到字符串首尾两个字符,判断是不是相等。如果不相等,那该字符串肯定不是对称的。否则我们接着判断里面的两个字符是不是相等,以此类推。基于这个思路,我们不难写出如下代码:

// Whether a string between pBegin and pEnd is symmetrical?

bool IsSymmetrical(char* pBegin, char* pEnd)

{

       if(pBegin == NULL || pEnd == NULL || pBegin > pEnd)

              return false;

 

       while(pBegin < pEnd)

       {

              if(*pBegin != *pEnd)

                     return false;

 

              pBegin++;

              pEnd --;

       }

 

       return true;

}

要判断一个字符串pString是不是对称的,我们只需要调用IsSymmetrical(pString, &pString[strlen(pString) – 1])就可以了。

解法一:On3)的算法

现在我们试着来得到对称子字符串的最大长度。最直观的做法就是得到输入字符串的所有子字符串,并逐个判断是不是对称的。如果一个子字符串是对称的,我们就得到它的长度。这样经过比较,就能得到最长的对称子字符串的长度了。于是,我们可以写出如下代码:

// Get the longest length of its all symmetrical substrings

// Time needed is O(T^3)

int GetLongestSymmetricalLength_1(char* pString)

{

       if(pString == NULL)

              return 0;

 

       int symmeticalLength = 1;

 

       char* pFirst = pString;

       int length = strlen(pString);

       while(pFirst < &pString[length - 1])

       {

              char* pLast = pFirst + 1;

              while(pLast <= &pString[length - 1])

              {

                     if(IsSymmetrical(pFirst, pLast))

                     {

                           int newLength = pLast - pFirst + 1;

                           if(newLength > symmeticalLength)

                                  symmeticalLength = newLength;                         

                     }

 

                     pLast++;

              }

 

              pFirst++;

       }

 

       return symmeticalLength;

}

我们来分析一下上述方法的时间效率。由于我们需要两重while循环,每重循环需要On)的时间。另外,我们在循环中调用了IsSymmetrical,每次调用也需要On)的时间。因此整个函数的时间效率是On3)。

通常On3)不会是一个高效的算法。如果我们仔细分析上述方法的比较过程,我们就能发现其中有很多重复的比较。假设我们需要判断一个子字符串具有aAa的形式(AaAa的子字符串,可能含有多个字符)。我们先把pFirst指向最前面的字符a,把pLast指向最后面的字符a,由于两个字符相同,我们在IsSymtical函数内部向后移动pFirst,向前移动pLast,以判断A是不是对称的。接下来若干步骤之后,由于A也是输入字符串的一个子字符串,我们需要再一次判断它是不是对称的。也就是说,我们重复多次地在判断A是不是对称的。

造成上述重复比较的根源在于IsSymmetrical的比较是从外向里进行的。在判断aAa是不是对称的时候,我们不知道A是不是对称的,因此需要花费On)的时间来判断。下次我们判断A是不是对称的时候,我们仍然需要On)的时间。

    测试代码为:
#include "stdafx.h"  
#include <iostream>  
using namespace std;  
  
 /************************************************  
 * 判断一个字符串是否是对称的  
 *************************************************/  
  
bool IsSymmetricalString(char* pstart,char* pend)  
{  
    if (pstart==NULL || pend==NULL || pstart>pend)  
    {  
        return false;  
    }  
    while (pstart<pend)  
    {  
        if (*pstart!=*pend)  
        {  
            return false;  
        }  
        pstart++;  
        pend--;  
    }  
    return true;  
}  
  
 /*************************************************  
 * 求最大对称子串的长度  
 **************************************************/  
int MaxSymmetricalSubstringLenth(char *pstring)  
{  
    if (pstring==NULL)  
    {  
        return 0;  
    }  
    int maxLength=1;  
    int length=strlen(pstring);  
    char* pstart=pstring;  
    while (pstart<&pstring[length-1])  
    {  
        char* psecond=pstart+1;  
        while (psecond<=&pstring[length-1])  
        {  
            if (IsSymmetricalString(pstart,psecond))  
            {  
                int tempLength=psecond-pstart+1;  
                if (tempLength>maxLength)  
                {  
                    maxLength=tempLength;  
                }  
            }  
            psecond++;  
        }  
        pstart++;  
    }  
    return maxLength;  
}  
  
  
int _tmain(int argc, _TCHAR* argv[])  
{  
    cout<<"Please Enter Your String(the length must <1000 ):"<<endl;  
    char* yourstring = new char[1000];  
    cin>>yourstring;  
    cout<<"The Max Symmetrical SubString Length in Your String is:"<<endl;  
    cout<<MaxSymmetricalSubstringLenth(yourstring)<<endl;  
  
  
    system("pause");  
    return 0;  
}  

解法二:On2)的算法

如果我们换一种思路,我们从里向外来判断。也就是我们先判断子字符串A是不是对称的。如果A不是对称的,那么向该子字符串两端各延长一个字符得到的字符串肯定不是对称的。如果A对称,那么我们只需要判断A两端延长的一个字符是不是相等的,如果相等,则延长后的字符串是对称的。因此在知道A是否对称之后,只需要O1)的时间就能知道aAa是不是对称的。

我们可以根据从里向外比较的思路写出如下代码:

// Get the longest length of its all symmetrical substrings

// Time needed is O(T^2)

int GetLongestSymmetricalLength_2(char* pString)

{

       if(pString == NULL)

              return 0;

 

       int symmeticalLength = 1;

      

       char* pChar = pString;

       while(*pChar != '\0')

       {

              // Substrings with odd length

              char* pFirst = pChar - 1;

              char* pLast = pChar + 1;

              while(pFirst >= pString && *pLast != '\0' && *pFirst == *pLast)

              {

                     pFirst--;

                     pLast++;

              }

 

              int newLength = pLast - pFirst - 1;

              if(newLength > symmeticalLength)

                     symmeticalLength = newLength;

 

              // Substrings with even length

              pFirst = pChar;

              pLast = pChar + 1;

              while(pFirst >= pString && *pLast != '\0' && *pFirst == *pLast)

              {

                     pFirst--;

                     pLast++;

              }

 

              newLength = pLast - pFirst - 1;

              if(newLength > symmeticalLength)

                     symmeticalLength = newLength;

 

              pChar++;

       }

 

       return symmeticalLength;

}

由于子字符串的长度可能是奇数也可能是偶数。长度是奇数的字符串是从只有一个字符的中心向两端延长出来,而长度为偶数的字符串是从一个有两个字符的中心向两端延长出来。因此我们的代码要把这种情况都考虑进去。

在上述代码中,我们从字符串的每个字符串两端开始延长,如果当前的子字符串是对称的,再判断延长之后的字符串是不是对称的。由于总共有On)个字符,每个字符可能延长On)次,每次延长时只需要O1)就能判断出是不是对称的,因此整个函数的时间效率是On2)。

测试代码:
// MaxSymmetricalSubstringLenth.cpp : 定义控制台应用程序的入口点。  
//  
  
#include "stdafx.h"  
  
#include<iostream>  
 #include<string>  
 using namespace std;  
   
 /*********************************************************************  
 * 计算字符串最大对称子串的长度  
 *********************************************************************/  
  
 int GetLongestSymmetricalLength_2(char* pString)  
 {  
     if(pString == NULL)  
         return 0;  
     int symmeticalLength = 1;  
     char* pChar = pString;  
     while(*pChar != '\0')  
     {  
         // Substrings with odd length(奇数)  
         char* pFirst = pChar - 1;  
         char* pLast = pChar + 1;  
         while(*pLast != '\0' && *pFirst == *pLast)  
         {  
             pFirst--;  
             pLast++;  
         }  
         int newLength = pLast - pFirst - 1;  
         if(newLength > symmeticalLength)  
             symmeticalLength = newLength;  
         // Substrings with even length(偶数)  
         pFirst = pChar;  
         pLast = pChar + 1;  
         while(  *pLast != '\0' && *pFirst == *pLast)  
         {  
             pFirst--;  
             pLast++;  
         }  
         newLength = pLast - pFirst - 1;  
         if(newLength > symmeticalLength)  
             symmeticalLength = newLength;  
         pChar++;  
     }  
     return symmeticalLength;  
  
 }  
  
  
   
 int main()  
 {  
     cout<<"Please Enter Your String:"<<endl;  
     char *yourstring = new char[1000];  
     cin>>yourstring;  
   
     cout<<"The Max Symmetrical SubString Length is:"<<endl;  
     cout<<GetLongestSymmetricalLength_2(yourstring)<<endl;  
   
     delete[] yourstring;   
  
     system("pause");  
     return 0;  
  
}  
参考来源:  http://zhedahht.blog.163.com/blog/static/25411174201063105120425/
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

程序员面试题精选100题(46)-对称子字符串的最大长度 的相关文章

  • 【超详细】树莓派4B 英特尔神经棒2代 Openvino安装记录

    主要参考了英特尔官方文档https docs openvinotoolkit org 2019 R3 1 docs install guides installing openvino raspbian html 还有同济子豪兄在达尔文的b
  • 2021-08-19-leetcode-00001

    二分查找 704 给定一个 n 个元素有序的 xff08 升序 xff09 整型数组 nums 和一个目标值 target xff0c 写一个函数搜索 nums 中的 target xff0c 如果目标值存在返回下标 xff0c 否则返回
  • FreeRTOS系统开发指南【精简版】

    文件说明 该文件主要根据FreeRTOS系统的功能 对FreeRTOS系统应用API函数进行项目开发进行指导和快速阅览 方便用户通过该文件快速使用FreeRTOS的内部资源来进行项目开发 其中涉及任务 时间管理 队列 信号量 定时器 内存管
  • 立创EDA学习笔记(3)——PCB绘制

    使用更新 转换原理图到PCB xff0c 将原理图导入PCB后 xff0c 点击工程中的PCB切换到该页面 xff0c 接下来进行PCB绘制 目录 一 放置板框 二 绘制定位孔 三 布局 四 布线 五 修改位号丝印大小 六 添加丝印 七 添
  • A star算法在三维避障路径规划的应用

    A star算法在三维避障路径规划的应用 前言 前言 在实际工程应用中 运动目标的外形 大小直接影响到路径选择 针对三维复杂场景的碰撞检测和路径规划问题 提出了一种基于层次包围盒碰撞检测的实时路径规划优化算法 该优化算法在进行碰撞检测时 通
  • Jetson TX2零基础学习(一)——连线、刷机

    目录 一 背景介绍 二 连线 三 刷机 四 鸣谢 五 结束语 系列文章 一 背景介绍 大家好 xff0c 我是潇湘小硕士 xff0c 注册账号已经两年有余 xff0c 今天第一次发文 xff0c 希望能够帮助到大家 我是通信专业研一学生一枚
  • 嵌入式Linux C多任务编程(进程篇)

    这俩天刚整理完进程部分内容 xff0c 再做个一个总结以便后期回顾 1 什么是多任务 xff1f 单任务vs多任务 单任务 xff1a 一个任务执行结束才能执行下一个任务 xff0c 或者说在一个任务执行得过程中不能响应其他任务 xff0c
  • 如何在Ubuntu上安装Boost

    本文翻译自 xff1a How to install Boost on Ubuntu I 39 m on Ubuntu and I want to install Boost 我在Ubuntu上 xff0c 并且想安装Boost I tri
  • A D 20:基于S T M 32的DDS信号源设计

    直接数字频率合成 xff08 DDS xff09 xff1a 根据正弦函数的产生原理 xff0c 直接对输入参考时钟进行抽样 数字化 xff0c 从相位出发 xff0c 用不同的相位给出不同的电压幅度 xff0c 最后经滤波平滑输出所需的频
  • 【STM32/FreeRTOS】SysTick定时器及FreeRTOS系统节拍

    目录 一 SysTick定时器 1 SysTick寄存器介绍 xff08 1 xff09 控制及状态寄存器 xff08 2 xff09 重装载数值寄存器 xff08 3 xff09 当前数值寄存器 2 SysTick寄存器配置函数 二 Fr
  • 【FreeRTOS】任务调度与任务切换

    目录 一 任务调度 二 任务切换 三 关于PendSV 一 任务调度 在创建好任务函数后 xff0c 需要调用函数vTaskStartScheduler 开启任务调度器 xff0c 创建的任务在调度器的调度下执行 开启任务调度器函数为 xf
  • k8s —— pod、init 容器、及资源清单的使用

    k8s pod init 容器 及资源清单的使用 文章目录 k8s pod init 容器 及资源清单的使用podPod生命周期init 容器 资源清单查询帮助文档 实验docker 镜像批量操作k8s 常用命令pod 资源清单init 容
  • TIVA Launchpad编程解锁好盈天行者(20A)电调

    电调解锁方法 2ms高电平的pwm波 xff0c 400hz xff0c 持续5s以上 1ms高电平的pwm波 xff0c 持续2s 即可解锁 xff0c 之后输入1 2ms范围的高电平的pwm波即可控制电机的转速 这里是主函数 span
  • C++第五次上机实验总结(加深对类和对象的理解)

    实验目的 xff1a 进一步加深对类和对象的理解 掌握集中对象传递的使用方法 掌握静态成员的概念和使用 实验共分成三部分 xff0c 分别为part a part b part c 实验内容 xff1a part a 了解三种不同对象传递方
  • C++第六次上机实验总结

    一 实验目的 xff1a 掌握派生类的声明方法和派生类构造函数的定义方法 xff1b 掌握不同方式下 xff0c 基类成员在派生类中的访问属性和访问规则 xff1b 二 程序代码 xff1a h文件 include lt iostream
  • C++第八次上机实验总结(多态)

    一 实验目的 xff1a 掌握C 43 43 语言多态性的基本概念 xff1b 掌握运算符重载函数的声明和定义方式 xff1b 二 试验任务 xff1a 1 编写一个程序 xff0c 实现两个负数相加 xff08 分别用类外定义运算符重载函
  • 机器学习实战之k-近邻算法(6)---手写数字识别系统(0-9识别)

    from numpy import import operator from os import listdir 创建数据集 def createDataSet group 61 array 1 0 1 1 1 0 1 0 0 0 0 0
  • termux—手机远程连接服务器教程

    文章目录 下载安装换源安装ssh软件连接服务器参考 下载安装 官网 可以从google play store下载安装 xff0c 也可从github上下载安装最新版本 app图标 安装完成后 xff0c 一些基础操作可以参考Termux 高
  • VScode主题色更换

    最新版VScode主题色更换 宝 xff0c 你是否有觉得它默认的黑色有点太晃眼 xff0c 想要拥有一个绿色或者浅色调的主题色呢 xff1f 当你想更换的时候你上网搜了很多 xff0c 发现版本不一样就是很迷惑呢 xff1f 让薇语帮你解
  • ROS学习笔记(1)ROS安装(推荐使用鱼香ROS安装工具,少走很多弯路)

    ROS安装 后记 xff1a 提前说一下 xff0c 按照网上的大部分ROS安装教程你会在下面的第四步遇到问题 xff0c 然后在网上找各种解决办法 xff0c 运气好的话你会很快解决 xff0c 但是也可能卡住半天没解决 xff08 比如

随机推荐

  • ubuntu18.04运行LiLi-OM

    一 上github下代码 https github com KIT ISAS lili om 1 1安装gtsam4 0 链接 xff1a GTSAM GTSAM is a BSD licensed C 43 43 library that
  • 软件KEIL串口应用-- printf调试

    KEIL5 里面实现printf的功能 xff0c 需要修改一个函数 重写 xff1a fputc 包含头文件 在当前 c文件中包含这两个头文件 重写函数 首先从原理图判断单片机芯片与上位机 xff08 电脑 xff09 通信是通过那个串口
  • 构造函数后加:符号

    分为三种情况 class animal public animal cout gt gt 34 animal 34 class fish public animal public fish animal cout gt gt 34 fish
  • 在gitee上新建仓库,将本地项目上传到新建的gitee仓库中

    1 首先登录gitee xff0c 点击右上角 43 号 xff0c 选择新建仓库 2 输入仓库名称及仓库简介 xff0c 选择是否开源 xff0c 下方的三个可不选 3 点击添加后 xff0c 页面如图所示 注 xff1a 此时gitee
  • L8Linux应用开发综合实战-在线词典项目(day1、2、3)

    目录 一 在线词典项目介绍及框架搭建 一 有道词典流程分析及本项目在线词典介绍 1 有道词典功能分析 2 项目流程 二 在线词典项目演示 三 流程示意图分析 1 客户端 2 服务器 四 客户端代码框架搭建 五 服务器端代码框架搭建 模板结构
  • matlab灰度图转化及二值化

    matlab灰度图转化及二值化 matlab提供图像处理功能 xff0c 我们可将彩色图像灰度化 xff0c 并对其进行二值化处理 xff0c 其简要代码如下 xff1a i span class token operator 61 spa
  • Docker - 编译安装nginx镜像

    目录 知识点1 xff1a 制作镜像的常用指令 RUN和CMD ENTRYPOINT的区别 首先需要一个安装nginx的脚本 制作Dockerfile 开始制作镜像 查看镜像是否制作成功 启动一个容器来测试镜像 编译安装ngixn镜像升级版
  • 阿克曼结构移动机器人的gazebo仿真(五)

    阿克曼结构移动机器人的gazebo仿真 xff08 五 xff09 第四章 用xacro优化URDF并配置gazebo仿真插件 0 前言 上节用简易模型写了一个小车的URDF代码 xff0c 这一节将用xacro对其进行优化 xff0c 这
  • 百度2014校园招聘笔试题武汉站三道算法设计题

    百度2014校园招聘笔试题武汉站三道算法设计题 1 给定任意一个整整数 求比这个数大且最小的不重复数 就是相邻两位不同 xff0c 例如1231 如1101就是重复数 解 xff1a 思路 xff1a 每次将给定的值加上1 xff0c 然后
  • gazebo版本升级以及环境太暗的解决方法

    gazebo升级 使用下列代码可将gazebo升级为该版本的最新版 xff0c 适用于gazebo7与gazebo9 添加源 sudo sh c 39 echo 34 deb http packages osrfoundation org
  • (3分钟了解)SLAM后端优化的四大金刚!g2o ceres gtsam SE-Sync

    后端优化常用的库有g2o ceres gtsam 和 se sync 这篇博客首先介绍se sync xff0c 然后比较四种库之间的差异 编辑切换为居中 添加图片注释 xff0c 不超过 140 字 xff08 可选 xff09 编辑切换
  • 基于Adams联合MATLAB的联合仿真设置

    因为最近在做一个四足机器人的仿真在网上找了一些资料基本上都不是说得很明白 下面是我参考了一些资料自己做的一个项目和对一些细节做的总结 xff0c 希望对大家有所帮助和解惑 本次联合仿真用到的软件主要是这三个Solidworks2018 Ad
  • MSC_ LICENSE. FILE = D:ladamsMAGNTUDElicense .dat

    ADAMS一段时间不使用后重新打开出现 解决办法 xff1a 1 找到原下载解压后的文件目录 2 点击MSC Calc 20161130 exe按照提示重新生成license dat文件 3 复制新的license dat文件到之前安装AD
  • 两块STM32F1之间互相通信(串口)

    首先准备两块STM32F103的板子 xff0c 以我这个为例 xff0c 我准备了一块STM32F103和CH32F103最小系统板子 xff0c 其他杜邦线 下载器及接线方法以及通信原理不再多说 这里我用的是STM32F103最小系统发
  • STM32单片机与蓝牙模块HC-05通信数据帧处理

    本章将会详细讲述蓝牙模块 xff08 HC 05 xff09 和STM32单片机之间的通信收发的数据如何处理 xff0c 在测试开始前首先在手机上下载好一个蓝牙调试APP xff0c 此APP可以是手机端和PC端口的 xff0c 以我常用的
  • N32G031固件库开发(三)基本TIM6定时器中断

    基本定时器 TIM6 基本定时器简介 基本定时器 TIM6 包含一个 16 位自动装载计数器 基本定时器主要特性 16位自动重载向上计数计数器 16位可编程预分频器 xff08 分频系数可配置为 1到 65536之间的任意值 xff09 产
  • N32G031固件库开发(四)通用定时器TIM3----PWM输出

    4 通用定时器 xff08 TIM3 xff09 通用定时器 xff08 TIM3 xff09 主要用于以下场合 xff1a 对输入信号进行计数 测量输入信号的脉冲宽度和产生输出波形等 4 1 TIM3 主要特性 16 位自动装载计数器 x
  • N32G031固件库开发(五)高级定时器TIM1----PWM输出

    高级控制定时器 xff08 TIM1 和 和 TIM8 xff09 5 1 TIM1 和 和 TIM8 简介 高级控制定时器 xff08 TIM1 和 TIM8 xff09 主要用于以下场合 xff1a 对输入信号进行计数 测量输入信号的脉
  • Java并发之semaphore(信号量)

    文章目录 1 官方解读2 通俗易懂的例子解析3 代码解析4 Semaphore的应用5 类结构和相关方法 1 类结构 2 acquire 方法 3 release 方法 6 总结 1 官方解读 semaphore信号量就是并发工具类 Sem
  • 程序员面试题精选100题(46)-对称子字符串的最大长度

    程序员面试题精选100题 46 xff0d 对称子字符串的最大长度 题目 xff1a 输入一个字符串 xff0c 输出该字符串中对称的子字符串的最大长度 比如输入字符串 google xff0c 由于该字符串里最长的对称子字符串是 goog