链表的回文结构(链表的中间节点+反转链表)

链表的回文结构

  • 一.链表的中间节点
    • 思路1:暴力求解
    • 思路2:快慢指针
  • 二.返回倒数第k个节点
    • 思路1:暴力求解
    • 思路2:快慢指针
  • 三.反转链表
    • 思路1:头插法
    • 思路2:反转指针的指向
  • 四.链表的回文结构
    • 思路1:利用数组,判断是否回文
    • 思路2:求链表的中间节点+反转链表

要解决链表的回文结构:首先需要求中间节点,其次是会反转链表。

一.链表的中间节点

链表的中间节点
在这里插入图片描述

思路1:暴力求解

  1. 求出链表的长度。
  2. 求出要返回的中间节点的位置(除2+1),遍历链表返回节点指针即可。
  3. 注意:兼容奇数个节点与偶数个节点。
typedef struct ListNode ListNode;

struct ListNode* middleNode(struct ListNode* head) 
{
    ListNode* cur = head;
    int listLength = 0; 
    while(cur)
    {
        //求链表的长度
        listLength++;
        cur = cur->next;
    }
    //链表中间节点的位置
    int middle = listLength / 2 + 1;
    int i = 1; //注意:非i=0
    cur = head;
    while(i < middle)
    {
        i++;
        cur = cur->next;
    }
    return cur;
}

思路2:快慢指针

  1. 定义两个指针fast、slow保存链表头节点的地址。
  2. 进入循环,fast指针一次走两个节点,slow指针一次走一个节点,当fast != NULL && fast->next != NULL时循环继续,否则循环结束。

情况1.含有奇数个节点
在这里插入图片描述

情况2.含有偶数个节点

在这里插入图片描述

typedef struct ListNode ListNode;

struct ListNode* middleNode(struct ListNode* head)
{
    //快慢指针:慢指针一次走一步,快指针一次走两步
    ListNode* fast = head;
    ListNode* slow = head;
    //注意循环继续的条件是&&而不是||,且fast与fast->next的位置不能交换
    while (fast != NULL && fast->next != NULL)
    {
        fast = fast->next->next;
        slow = slow->next;
    }
    return slow;
}

二.返回倒数第k个节点

返回倒数第k个节点

在这里插入图片描述

思路1:暴力求解

  1. 遍历链表求链表的长度length
  2. 倒数第k个节点,等价于从前往后的第length - k个节点。
  3. 再次遍历链表找到第length - k个节点,返回节点指针即可。

在这里插入图片描述

typedef struct ListNode ListNode;

int kthToLast(struct ListNode* head, int k)
{
    //1.遍历链表求出链表长度,再遍历一次链表,找到返回值
	int size = 0;
    ListNode* cur = head;
    while(cur)
    {
        size++;
        cur = cur->next;
    }
    int i = 0;
    cur = head;
    while(i < size - k)
    {
        cur = cur->next;
        i++;
    }
    return cur->val;
}

思路2:快慢指针

  1. 定义两个指针fast、slow保存链表头节点的地址。
  2. fast指针先走k个节点
  3. 进入循环,fast与slow指针各自每次走一个节点,当fast != NULL时循环继续,否则循环结束。

在这里插入图片描述

typedef struct ListNode ListNode;

int kthToLast(struct ListNode* head, int k)
{
	//2.快慢指针:快指针先走k步,然后快指针一次走一步,慢指针一次走一步
    ListNode* fast = head;
    ListNode* slow = head;

    for (int i = 0; i < k; i++)
    {
        fast = fast->next;
    }
    while (fast != NULL)
    {
        fast = fast->next;
        slow = slow->next;
    }
    return slow->val;
}

三.反转链表

反转链表

思路1:头插法

  1. 创建新链表 newHead = NULL。
  2. 遍历原链表,逐个节点头插倒新链表中。

在这里插入图片描述

typedef struct ListNode ListNode;

struct ListNode* reverseList(struct ListNode* head) 
{
    //1.创建新链表,遍历原链表,逐个头插
    ListNode* newHead = NULL, *cur = head;
    while(cur)
    {
        //头插
        ListNode* next = cur->next;
        cur->next = newHead;
        newHead = cur;
        cur = next;
    }
    return newHead;
}

思路2:反转指针的指向

在这里插入图片描述

typedef struct ListNode ListNode;

struct ListNode* reverseList(struct ListNode* head) 
{
	//2.创建三个指针,反转指针的指向
    if(head == NULL)
    {
        return NULL;
    }
    ListNode* n1 = NULL, *n2 = head, *n3 = n2->next;
    while(n2)
    {
        n2->next = n1;
        n1 = n2;
        n2 = n3;
        if(n3 != NULL)
        {
            n3 = n3->next;
        }   
    }
    return n1;
}
    

四.链表的回文结构

链表的回文结构

思路1:利用数组,判断是否回文

class PalindromeList {
public:
    //判断数组是否满足回文结构
    bool isReverse(int arr[], int left, int right)
    {
        while(left < right)
        {
            if(arr[left] != arr[right])
            {
                return false;
            }
            left++;
            right--;
        }
        return true;
    }
    bool chkPalindrome(ListNode* A)
    {
        int arr[900];
        ListNode* cur = A;
        int i = 0, listLength = 0;
        while(cur)
        {
            arr[i++] = cur->val;//将链表中的值保存到数组中
            cur = cur->next;
            listLength++;//求链表的长度
        }
        return isReverse(arr, 0, listLength - 1);
    }
};

思路2:求链表的中间节点+反转链表

  1. 寻找链表的中间节点 mid。
  2. 将中间节点 mid 以及之后的节点组成的链表反转。
  3. 遍历反转后的链表,当一个一个与原链表的数据域对比,若相同则是回文结构。

情况1.含有奇数个节点:
在这里插入图片描述
情况2.含有偶数个节点:

在这里插入图片描述

class PalindromeList {
public:
    ListNode* findMidNode(ListNode* phead)
    {
        ListNode* fast = phead;
        ListNode* slow = phead;
        while(fast && fast->next)
        {
            slow = slow->next;
            fast = fast->next->next;
        }
        return slow;
    }
    ListNode* reverseList(ListNode* phead)
    {
        ListNode* n1, *n2, *n3;
        n1 = NULL, n2 = phead, n3 = n2->next;
        while(n2)
        {
            n2->next = n1;
            n1 = n2;
            n2 = n3;
            if(n3 != NULL)
            {
                n3 = n3->next;
            }         
        }
        return n1;
    }
    bool chkPalindrome(ListNode* A) 
    {
        //1.找链表的中间节点
        ListNode* mid = findMidNode(A);
        //2.反转中间节点以及之后的节点组成的链表
        ListNode* right = reverseList(mid);
        //3.遍历反转链表,与原链表进制值的比较
        ListNode* left = A;
        while(right)
        {
            if(right->val != left->val)
            {
                return false;
            }
            right = right->next;
            left = left->next;
        }
        return true;
    }
};

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

react + redux 状态管理操作

目录 1 概念2 Redux 安装3 创建子模块并导入4 中间件为 react 注入 store5 在组件中使用 store 数据6 修改 store 数据7 提交 action 传参8 异步状态操作9 redux 调试工具 1 概念 Redux 是一个全局状态管理的 JS 库 2 Redux 安装 在react中使用redux&#xff0c;官方要求安…

css设置弹性flex后,如果设置100vh高度不撑满的原因

问题 父元素设置height为100%&#xff0c;有两个子元素&#xff0c;第一个设置height:100vh&#xff0c;第二个设置flex:1&#xff0c;此时第一个高度无法撑满盒子 原因解决方式 当父元素设置display为flex,第一个div设置高度64px,剩一个div设置高度为flex&#xff1a;1,这时…

DROO论文笔记

推荐文章DROO源码及论文学习 读论文《Deep Reinforcement Learning for Online Computation Offloading in Wireless Powered Mobile-Edge Computing Networks》的笔记 论文地址&#xff1a;用于无线移动边缘计算网络在线计算卸载的深度强化学习 论文代码地址&#xff1a;DR…

AG32 的MCU与FPGA的主频可以达到568MHz吗

Customers: AG32/ AGRV2K 这个芯片主频和定时器最高速度是多少&#xff1f;用户期望 CPLD计时器功能0.1ns以下。 AGM RE: CPLD做不到 0.1ns的速率&#xff0c;这个需要10G以上的时钟。 那AGRV2K最高多少MHz呢&#xff1f; 一般200MHZ比较容易实现。 进一步说明&#xff1…

Vulnhub靶场DC-3-2练习

目录 0x00 准备0x01 主机信息收集0x02 站点信息收集0x03 漏洞查找与利用1. joomla漏洞查找2. SQL注入漏洞3. 破解hash4. 上传一句话木马5. 蚁剑连接shell6. 反弹shell7. 提权 0x04 总结 0x00 准备 下载链接&#xff1a;https://download.vulnhub.com/dc/DC-3-2.zip 介绍&#…

51单片机5(GPIO简介)

一、序言&#xff1a;不论学习什么单片机&#xff0c;最简单的外设莫过于I口的高低电平的操作&#xff0c;接下来&#xff0c;我们将给大家介绍一下如何在创建好的工程模板上面&#xff0c;通过控制51单片机的GPIO来使我们的开发板上的LED来点亮。 二、51单片机GPIO介绍&#…

数据结构初阶(C语言)-复杂度的介绍

在学习顺序表之前&#xff0c;我们需要先了解下什么是复杂度&#xff1a; 一&#xff0c;复杂度的概念 我们在进行代码的写作时&#xff0c;通常需要用到许多算法&#xff0c;而这些算法又有优劣之分&#xff0c;区分算法的优劣则是通过算法的时间复杂度和空间复杂度来决定。 …

python 怎样生成窗体

通过import tkinter导入Tkinter模块&#xff0c;没有这句下面的都不成立了。 wintkinter.Tk()&#xff0c;这句是创建windows的窗口对象&#xff0c;注意后面的Tk&#xff0c;大小写。 win.title("窗口")&#xff0c;这段是设置窗口上的标题。 另外窗口的大小你可以通…

Linux命令更新-Vim 编辑器

简介 Vim 是 Linux 系统中常用的文本编辑器&#xff0c;功能强大、可扩展性强&#xff0c;支持多种编辑模式和操作命令&#xff0c;被广泛应用于程序开发、系统管理等领域。 1. Vim 命令模式 Vim 启动后默认进入命令模式&#xff0c;此时键盘输入的命令将用于控制编辑器本身&…

云计算【第一阶段(31)】PXE高效批量网络装机

一、系统安装 1.1、系统装机的三种引导方式 1. 硬盘 2. 光驱&#xff08; u 盘&#xff09; 3. 网络启动 pxe 1.2、系统安装过程 加载boot loader Boot Loader 是在操作系统内核运行之前运行的一段小程序。通过这段小程序&#xff0c;我们可以初始化硬件设备、建立内存空间的映…

解决mysql,Navicat for MySQL,IntelliJ IDEA之间中文乱码

使用软件版本 jdk-8u171-windows-x64 ideaIU-2021.1.3 mysql-essential-5.0.87-win32 navicat8_mysql_cs 这个问题我调试了好久&#xff0c;网上的方法基本上都试过了&#xff0c;终于是解决了。 三个地方结果都不一样。 方法一 首先大家可以尝试下面这种方法&#xff1a…

无人驾驶大热,新能源汽车智能化中的算网支持

来源新华社&#xff1a;百度“萝卜快跑”全无人驾驶汽车行驶在路上 当前&#xff0c;新能源汽车产业数智化已成为全球汽车产业数字化转型的焦点。一方面&#xff0c;随着人工智能、大数据、云计算等技术的深度融合&#xff0c;新能源汽车在自动驾驶、智能互联、能源管理等方面…

【自动驾驶汽车通讯协议】UART通信详解:理解串行数据传输的基石

文章目录 0. 前言1. 同步通讯与异步通讯1.1 同步通信1.2 异步通信 2. UART的数据格式3. 工作原理3.1 波特率和比特率3.2 UART的关键特性 4. UART在自动驾驶汽车中的典型应用4.1 UART特性4.2应用示例 5. 结语 0. 前言 按照国际惯例&#xff0c;首先声明&#xff1a;本文只是我自…

STM32MP135裸机编程:BOOT跳转到APP前关闭所有中断、清除所有中断挂起标志操作方法

0 前言 一般来说&#xff0c;MCU/SOC的BOOT在跳转到APP前都需要进行环境清理的操作&#xff0c;其中必须进行的一项操作便是关闭所有中断、清除所有中断挂起标志。本文介绍基于STM32MP135裸机编程下关闭所有中断、清除所有中断挂起标志的操作方法。 1 操作方法 STM32MP135裸…

关于Kafka Topic分区和Replication分配的策略

文章目录 1. Topic多分区2. 理想的策略3. 实际的策略4. 如何自定义策略 1. Topic多分区 如图&#xff0c;是一个多分区Topic在Kafka集群中可能得分配情况。 P0-RL代表分区0&#xff0c;Leader副本。 这个Topic是3分区2副本的配置。分区尽量均匀分在不同的Broker上&#xff0c…

怎么减少pdf的MB,怎么减少pdf的大小

在数字化时代&#xff0c;pdf文件因其格式稳定、跨平台兼容性强等特点而广受欢迎。然而&#xff0c;随着内容的丰富&#xff0c;pdf文件的大小也日益增大&#xff0c;给文件传输和存储带来了不少困扰。本文将为你介绍多种减小pdf文件大小的方法&#xff0c;帮助你轻松应对这一问…

【ChatGPT】深入解析Prompt提示词及如何高效使用ChatGPT

一、Prompt提示词是什么&#xff1f; 1.1 Prompt的定义 Prompt是人工智能领域中的一个关键概念&#xff0c;尤其在自然语言处理&#xff08;NLP&#xff09;和生成型AI模型中。简而言之&#xff0c;prompt是一段文本或指令&#xff0c;用于引导或启动AI模型的特定响应或操作。…

成为CMake砖家(2): macOS创建CMake本地文档的app

大家好&#xff0c;我是白鱼。 使用 CMake 的小伙伴&#xff0c; 有的是在 Windows 上&#xff0c; 还有的是在 macOS 上。之前咱们讲了 windows 上查看 cmake 本地 html 文档的方式&#xff0c; 这篇讲讲 macOS 上查看 cmake 本地 html 文档的方法。 1. 问题描述 当使用 CMa…

C1W1.LAB.Preprocessing+Word frequencies+Logistic_regression_model

理论课&#xff1a;C1W1.Sentiment Analysis with Logistic Regression 文章目录 预处理导入包Twitter dataset简介查看原始文本处理原始文本处理超链接、Twitter 标记和样式分词去除标点和停用词词干处理 process_tweet() 词频构建与可视化导入包加载数据集字典字典实例添加或…

什么是im即时通讯?WorkPlus im即时通讯私有化部署安全可控

IM即时通讯是Instant Messaging的缩写&#xff0c;指的是一种实时的、即时的电子信息交流方式&#xff0c;也被称为即时通讯。它通过互联网和移动通信网络&#xff0c;使用户能够及时交换文本消息、语音通话、视频通话、文件共享等信息。而WorkPlus im即时通讯私有化部署则提供…