链表的回文结构(详解)

2024-05-03 09:04
文章标签 链表 详解 回文结构

本文主要是介绍链表的回文结构(详解),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

链表的回文结构(详解)

题目:

链表的回文结构

对于一个链表,请设计一个时间复杂度为O(n),额外空间复杂度为O(1)的算法,判断其是否为回文结构。

给定一个链表的头指针A,请返回一个bool值,代表其是否为回文结构。保证链表长度小于等于900。

测试样例:

1->2->2->1
返回:true

思路:

  1. 我们想要通过对比链表的前一半和链表的后一半是否相同来判断是否为回文结构

  2. 那我们就要找到中间节点,也就是快慢指针法

  3. image-20240503004231573

  4. 不仅如此,我们还要通过prev来记录slow的前一个节点,这样我们在后面需要对prev进行断联,我们才得到该节点

  5. image-20240503004434010

  6. 这个时候将slow节点和之后的节点进行逆置,创建一个新链表,让slow节点和之后的节点依次头插到新链表当中

  7. image-20240503004915596

  8. 逆置完链表之后,要记得让slow节点之前的节点断联 ,也就是prev->next = NULL

  9. 然后就是遍历原链表 判断两个链表的val值是否相等,不相等返回false,相等返回true

image-20240503001805792

代码:

struct ListNode {int val;struct ListNode* next;
};typedef struct ListNode ListNode;
bool chkPalindrome(ListNode* A)
{// 创建快慢指针,去找到链表的中间节点ListNode* slow, * fast;slow = fast = A;// 创建一个prev指针去专门指向slow的前一个节点 最终找到的是中间节点的前一个节点ListNode* prev = A;// 找到中间节点  2slow = fastwhile (fast && fast->next){prev = slow; // 指向slow的前一个节点slow = slow->next;fast = fast->next->next;}// 此时slow就是A链表的中间节点// 这个时候我们逆置slow节点后面的节点// 如果和slow节点前面的节点一样,那就是回文结构// 创建新链表ListNode* newhead = NULL;// 将slow节点及以后的节点逆置// 也就是让后面的节点依次头插到新链表中ListNode* pcur = slow;while (pcur){ListNode* next = pcur->next; // 让next记载pcur的下一个节点pcur->next = newhead; // 让pcur头插到新链表中newhead = pcur; // 更新第一个节点pcur = next; // 让pcur继续向后遍历}// 走到这里 slow及slow后面的节点都逆置完毕了// 但是有一个需要注意的地方,我们的slow的前一个节点仍然指向的是slow节点// 但是我们又新创建了一个逆置链表,那么prev此时的next指针指向的就是逆置链表的尾节点// 因此我们需要将prev不在指向slow节点,而是NULLprev->next = NULL;// 这个时候我们去判断 新链表是否和 A链表第一个节点到slow节点之前相等ListNode* cur = A;while (cur){// 判断原链表和新链表的val值是否相同if (cur->val != newhead->val)return false; // 不同就返回falsecur = cur->next; //相同就继续往下走newhead = newhead->next;}// 走到这里说明是回文结构了return true;
}

这篇关于链表的回文结构(详解)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/956359

相关文章

Flutter监听当前页面可见与隐藏状态的代码详解

《Flutter监听当前页面可见与隐藏状态的代码详解》文章介绍了如何在Flutter中使用路由观察者来监听应用进入前台或后台状态以及页面的显示和隐藏,并通过代码示例讲解的非常详细,需要的朋友可以参考下... flutter 可以监听 app 进入前台还是后台状态,也可以监听当http://www.cppcn

C++ Primer 标准库vector示例详解

《C++Primer标准库vector示例详解》该文章主要介绍了C++标准库中的vector类型,包括其定义、初始化、成员函数以及常见操作,文章详细解释了如何使用vector来存储和操作对象集合,... 目录3.3标准库Vector定义和初始化vector对象通列表初始化vector对象创建指定数量的元素值

MyBatis与其使用方法示例详解

《MyBatis与其使用方法示例详解》MyBatis是一个支持自定义SQL的持久层框架,通过XML文件实现SQL配置和数据映射,简化了JDBC代码的编写,本文给大家介绍MyBatis与其使用方法讲解,... 目录ORM缺优分析MyBATisMyBatis的工作流程MyBatis的基本使用环境准备MyBati

Nginx中location实现多条件匹配的方法详解

《Nginx中location实现多条件匹配的方法详解》在Nginx中,location指令用于匹配请求的URI,虽然location本身是基于单一匹配规则的,但可以通过多种方式实现多个条件的匹配逻辑... 目录1. 概述2. 实现多条件匹配的方式2.1 使用多个 location 块2.2 使用正则表达式

spring @EventListener 事件与监听的示例详解

《spring@EventListener事件与监听的示例详解》本文介绍了自定义Spring事件和监听器的方法,包括如何发布事件、监听事件以及如何处理异步事件,通过示例代码和日志,展示了事件的顺序... 目录1、自定义Application Event2、自定义监听3、测试4、源代码5、其他5.1 顺序执行

Java之并行流(Parallel Stream)使用详解

《Java之并行流(ParallelStream)使用详解》Java并行流(ParallelStream)通过多线程并行处理集合数据,利用Fork/Join框架加速计算,适用于大规模数据集和计算密集... 目录Java并行流(Parallel Stream)1. 核心概念与原理2. 创建并行流的方式3. 适

web网络安全之跨站脚本攻击(XSS)详解

《web网络安全之跨站脚本攻击(XSS)详解》:本文主要介绍web网络安全之跨站脚本攻击(XSS)的相关资料,跨站脚本攻击XSS是一种常见的Web安全漏洞,攻击者通过注入恶意脚本诱使用户执行,可能... 目录前言XSS 的类型1. 存储型 XSS(Stored XSS)示例:危害:2. 反射型 XSS(Re

linux本机进程间通信之UDS详解

《linux本机进程间通信之UDS详解》文章介绍了Unix域套接字(UDS)的使用方法,这是一种在同一台主机上不同进程间通信的方式,UDS支持三种套接字类型:SOCK_STREAM、SOCK_DGRA... 目录基础概念本机进程间通信socket实现AF_INET数据收发示意图AF_Unix数据收发流程图A

Go 1.23中Timer无buffer的实现方式详解

《Go1.23中Timer无buffer的实现方式详解》在Go1.23中,Timer的实现通常是通过time包提供的time.Timer类型来实现的,本文主要介绍了Go1.23中Timer无buff... 目录Timer 的基本实现无缓冲区的实现自定义无缓冲 Timer 实现更复杂的 Timer 实现总结在

Vue中动态权限到按钮的完整实现方案详解

《Vue中动态权限到按钮的完整实现方案详解》这篇文章主要为大家详细介绍了Vue如何在现有方案的基础上加入对路由的增、删、改、查权限控制,感兴趣的小伙伴可以跟随小编一起学习一下... 目录一、数据库设计扩展1.1 修改路由表(routes)1.2 修改角色与路由权限表(role_routes)二、后端接口设计