代码随想录算法训练营第四天| 24. 两两交换链表中的节点、19.删除链表的倒数第N个节点 、 面试题 02.07. 链表相交、142.环形链表II

news2025/1/12 12:07:57

24. 两两交换链表中的节点

在这里插入图片描述

题目链接: 24. 两两交换链表中的节点
文档讲解:代码随想录
状态:没做出来,没有正确更新头节点,因为head和cur共享引用,会随着cur的移动,丢失之前存放的节点

错误代码:

    public ListNode swapPairs(ListNode head) {
        ListNode cur = head;
        ListNode next;
        ListNode temp;

        while (cur != null && cur.next != null) {
            next = cur.next;
            temp = next.next;

            next.next = cur;
            cur.next = temp;

            cur = temp;
        }

        return head;

    }

思路:前面博客中总结过啥时候需要使用虚拟头结点,这边需要返回head节点,所以使用dummy节点,然后cur从dummy出发。

public ListNode swapPairs(ListNode head) {
    // 创建一个虚拟节点,dummy.next 指向 head
    ListNode dummy = new ListNode();
    dummy.next = head;

    // cur 用于遍历链表,初始化为 dummy 节点
    ListNode cur = dummy;
    ListNode first;   // 用于指向要交换的第一个节点
    ListNode second;  // 用于指向要交换的第二个节点
    ListNode temp;    // 用于暂存第二个节点的下一个节点

    // 当 cur 后面至少有两个节点时,继续交换
    while (cur.next != null && cur.next.next != null) {
        first = cur.next;          // 定位要交换的第一个节点
        second = cur.next.next;    // 定位要交换的第二个节点
        temp = second.next;        // 暂存第二个节点的下一个节点

        // 进行节点交换
        first.next = temp;         // 第一个节点的 next 指向第二个节点的 next
        second.next = first;       // 第二个节点的 next 指向第一个节点
        cur.next = second;         // 当前节点的 next 指向第二个节点

        // cur 移动到已交换的两个节点之后的位置
        cur = first;
    }

    // 返回新的头节点
    return dummy.next;
}

19.删除链表的倒数第N个节点

在这里插入图片描述

题目链接: 19.删除链表的倒数第N个节点
文档讲解:代码随想录
状态:so easy

思路:
看到“返回链表的头结点”,使用虚拟头结点dummy,删除倒数第N个节点就需要先找到它,然后对它进行操作。
方式1:遍历一遍得到len,然后倒数第n个节点就是len-n+1个节点。
方式2:利用栈,先所有节点入栈,然后出栈n个节点,此时栈顶元素就是倒数第N+1个节点,让它的next节点指向下下个节点即可。
方式3:利用双指针中的快慢指针,让fast指针先走n步,然后和slow指针同时移动,当fast指针指向null时,slow指针指向倒数第N=1个节点,让它的next节点指向下下个节点即可。

双指针题解

public ListNode removeNthFromEnd(ListNode head, int n) {
        // 创建一个虚拟节点,dummy.next 指向 head
        ListNode dummy = new ListNode();
        dummy.next = head;
        // 初始化快指针和慢指针都指向虚拟节点
        ListNode fast = dummy;
        ListNode slow = dummy;

        // 让快指针先移动 n 步
        while (n-- > 0 && fast.next != null) {
            fast = fast.next;
        }

        // 同时移动快指针和慢指针,直到快指针到达链表末尾
        while (fast.next != null) {
            fast = fast.next;
            slow = slow.next;
        }

        // 慢指针的下一个节点就是要删除的节点
        slow.next = slow.next.next;

        // 返回新的头节点
        return dummy.next;
    }

面试题 02.07. 链表相交

在这里插入图片描述

题目链接: 面试题 02.07. 链表相交(同160.链表相交)
文档讲解:代码随想录
状态:没做出来,一个bug卡了很久。

错误代码:

    // 在每一轮循环中,指针先移动到下一个节点,然后才判断是否为null并进行重定向。这会导致在指针都为null时直接重定向,而没有机会检查pointerA == pointerB是否成立,导致死循环。
    public ListNode getIntersectionNode(ListNode headA, ListNode headB) {
        ListNode pointerA = headA;
        ListNode pointerB = headB;

        while (pointerA != pointerB) {
            pointerA = pointerA.next;
            pointerB = pointerB.next;

            if (pointerA == null) {
                pointerA = headB;
            }
            if (pointerB == null) {
                pointerB = headA;
            }
        }
        return pointerA;
    }

思路1:可以使用HashMap,headA中的节点都存在map中,遍历headB时检查是否在map中存在该节点。如果存在,则这个节点就是交点
思路2:双指针。pA遍历headA,pB遍历headB,当pA遍历到尾部时,重定向到headB,当pB遍历到尾部时,重定向到headA,如果存在相同节点,则会两个指针在同一个节点相遇。

双指针题解:

   /**
     * 法二:双指针
     * 指针 pA和pB 指向同一个节点或者都为空时,返回它们指向的节点或者 null
     * <p>这边指向相同节点的是8,不是5也不是1哦,前面的5和1只是值相等,地址不等
     * 4 1 8 4 5#5 0 1 8 4 5
     * 5 0 1 8 4 5#4 1 8 4 5
     */
    public ListNode getIntersectionNode(ListNode headA, ListNode headB) {
        // 初始化两个指针,分别指向两个链表的头节点
        ListNode pA = headA;
        ListNode pB = headB;

        // 遍历两个链表,直到找到交点或者遍历结束
        while (pA != pB) {
            // 如果当前指针不为空,则移动到下一个节点;否则,重定向到另一个链表的头部
            pA = (pA != null) ? pA.next : headB;
            pB = (pB != null) ? pB.next : headA;
        }

        // 返回交点的节点,如果没有交点则返回null
        return pA;
    }

142.环形链表II

在这里插入图片描述

题目链接: 142.环形链表II
文档讲解:代码随想录
状态:还行

思路1:遍历head节点,用HashSet检查是否有出现过的同一节点,如果没有,就将遍历到的节点存入HashSet,否则返回的节点就是环的入口节点
思路2:数学方法+快慢指针

快慢指针题解

 // 使用快慢指针法找到单链表中的环的入口点
    public ListNode detectCycle(ListNode head) {
        // 初始化快慢指针,初始位置为链表头部
        ListNode fast = head, slow = head;
        // 防止空链表或者不存在环的情况
        while (true) {
            // 如果快指针或者快指针的下一个节点为null,说明不存在环,返回null
            if (fast == null || fast.next == null) {
                return null;
            }
            // 快指针每次移动两步,慢指针每次移动一步
            fast = fast.next.next;
            slow = slow.next;
            // 如果快慢指针相遇,说明存在环,跳出循环
            if (fast == slow) {
                break;
            }
        }
        // 将快指针重新指向链表头部,慢指针不变
        fast = head;
        /*当快慢指针再次相遇时,即为环的入口点
          这是因为在快慢指针相遇时,慢指针走过的距离(从链表头到环入口点的距离)与快指针走过的距离之间存在一定的关系。假设链表头到环入口点的距离为A,快慢指针相遇点距离环入口点的距离为B,环的周长为C。
          当快慢指针相遇时,快指针已经在环内绕了n圈,所以慢指针走的距离是(A+ B),而快指针走的距离是(A+B+nC)。
          由于快指针是慢指针的两倍速度,因此有关系:快指针走过的距离是慢指针的两倍。可以得到以下方程:
          A+B+nC=2(A+B)
          从而可以推导出:
          A= (n-1)C+(C-B)
          这意味着,将快指针重新指向链表头部,然后慢指针和快指针以相同的速度前进,它们将在环的入口点相遇。
          这是因为,慢指针走了n—1圈的环,再走C—B的距离就到达了环的入口点,
          而根据公式,同样的速度快指针从链表头步出发,此时刚好也走了距离A,此时两个指针在入口点相遇
         */
        while (slow != fast) {
            slow = slow.next;
            fast = fast.next;
        }
        // 返回环的入口点
        return fast;
    }

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

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

相关文章

6.定时器分时复用测量占空比

1.CUBEMAX配置 测量PA6&#xff0c;PA7输出的占空比&#xff0c;只需要把主要的配置&#xff0c;配置为A6口就行&#xff0c;A7口黄色表示配置不正确&#xff0c;不用管。 2.软件代码 TIME.c中找到TIM3的初始化&#xff0c;在后面初始化A7口 void MX_TIM3_Init_PA7(void) {/*…

JVM之【类加载机制】

一、类加载过程 1. 加载&#xff08;Loading&#xff09; 工作内容&#xff1a; 通过类的全限定名来获取定义此类的二进制字节流&#xff1a; JVM首先会调用类加载器的findClass方法来找到类文件的路径&#xff0c;通常从文件系统、JAR包、网络、数据库等来源获取类文件。 将…

揭秘Markdown:轻松掌握基础语法,让你的写作更高效、优雅!

文章目录 前言1.标题1.1 使用 和 - 标记一级和二级标题1.2 使用 # 号标记 2.段落格式2.1 字体2.2 分割线2.3 删除线2.4 下划线2.5 脚注 3.列表3.1 无序列表3.2 有序列表3.3 列表嵌套 4.区块4.1 区块中使用列表4.2 列表中使用区块 5.代码代码区块 6.链接7.图片8.表格9.高级技巧…

C#对文件进行批量重命名或者对某个单独的文件进行改名

目录 一、FolderBrowserDialog 二、OpenFileDialog 三、Path 四、ui设计 五、代码部分 一、FolderBrowserDialog FolderBrowserDialog是一个用于选择文件夹的对话框控件&#xff0c;可以在windows Forms应用程序中使用。使用它可以让用户选择一个文件夹&#xff0c;并返…

闲话 .NET(6):.NET Core 各个版本的特性

前言 之前我们聊了一下 .NET Core 有哪些优势&#xff0c;.NET Core 发展非常迅速&#xff0c;不过短短几年&#xff0c;.NET Core 已经发布 .NET 8 了&#xff0c;基本上保持了一年一个版本的速度&#xff0c;每个版本都有自己的独有特性&#xff0c;下面我们来简单的盘点一下…

从零实现Llama3中文版

1.前言 一个月前&#xff0c;Meta 发布了开源大模型 llama3 系列&#xff0c;在多个关键基准测试中优于业界 SOTA 模型&#xff0c;并在代码生成任务上全面领先。 此后&#xff0c;开发者们便开始了本地部署和实现&#xff0c;比如 llama3 的中文实现、llama3 的纯 NumPy 实现…

【机器学习300问】100、怎么理解卷积神经网络CNN中的池化操作?

一、什么是池化&#xff1f; 卷积神经网络&#xff08;CNN&#xff09;中的池化&#xff08;Pooling&#xff09;操作是一种下采样技术&#xff0c;其目的是减少数据的空间维度&#xff08;宽度和高度&#xff09;&#xff0c;同时保持最重要的特征并降低计算复杂度。池化操作不…

高速数据采集与传输(一):ADC08D500调研

前言&#xff1a;高速ADC数据采集的应用和开发&#xff0c;涉及的技术面非常的广泛&#xff0c;后续阶段博主将尝试以纯项目开发的形式做一次专题技术分享&#xff0c;将基于高速数据采集的相关内容进行一系列的技术文档更新。博主全凭兴趣在更新和总结&#xff0c;很难做到一直…

AI预测体彩排3采取888=3策略+和值012路一缩定乾坤测试5月26日预测第2弹

今天继续基于8883的大底进行测试&#xff0c;昨天的预测已成功命中&#xff01;今天继续测试&#xff0c;按照排三前面的规律&#xff0c;感觉要出对子了&#xff0c;所以本次预测不再杀对子&#xff0c;将采用杀一个和尾来代替。好了&#xff0c;直接上结果吧~ 首先&#xff0…

软考结束。有什么要说的

1. 竟然是机试&#xff0c;出乎我意料。是 考试机构觉得笔试成本高了么。这次的考试是机试&#xff0c;相比以往有所不一样。感言是不是以后都会在固定地点考试也说不准。 2. 遇到年轻人。 这次旁边的一个女同学第一次参加&#xff0c;还像我询问了一些关于软考的事。我是有…

14 vue学习:透传Attributes

Attributes 继承 “透传 attribute”指的是传递给一个组件&#xff0c;却没有被该组件声明为 [props]或 [emits]的 attribute 或者 v-on 事件监听器。最常见的例子就是 class、style 和 id。 当一个组件以单个元素为根作渲染时&#xff0c;透传的 attribute 会自动被添加到根元…

vcpkg环境配置

vcpkg 使用linux相关库&#xff0c;设置环境变量VCPKG_ROOT&#xff0c;设置cmake工具链$VCPKG_ROOT/scripts\buildsystems\vcpkg.cmake set VCPKG_DEFAULT_TRIPLETx64-windows .\vcpkg.exe install fftw3 freetype gettext glibmm gtkmm libjpeg-turbo libpng libxmlpp libs…

pytest框架用例命名规则详解

pytest 测试用例的命名规则是为了确保 pytest 能够正确地识别和执行测试用例。 以下是关于 pytest 测试用例命名规则的详细解释&#xff1a; 1 单个测试文件以‘test_’开头或者以‘_test’结尾 比如我们创建test_case1.py case2_test.py文件。 2 单个测试文件中&#xff0c…

PLC_博图系列☞R_TRIG:检测信号上升沿

PLC_博图系列☞R_TRIG&#xff1a;检测信号上升沿 文章目录 PLC_博图系列☞R_TRIG&#xff1a;检测信号上升沿背景介绍R_TRIG&#xff1a; 检测信号上升沿说明参数示例 关键字&#xff1a; PLC、 西门子、 博图、 Siemens 、 R_TRIG 背景介绍 这是一篇关于PLC编程的文章&a…

服务器安全审计: chkrootkit 和 rkhunter 详解

chkrootkit 和 rkhunter 是两个广泛使用的安全工具,用于检测系统是否被Rootkit或其他恶意软件感染。本文将详细说明这两个工具的使用方法及如何解释检测结果。 1. chkrootkit 1.1. 安装 chkrootkit 在CentOS上安装 chkrootkit 可以使用以下命令: yum install chkrootkit -…

SpringMVC:创建一个简单的SpringMVC框架S

目录 一、框架介绍 两个重要的xml文件 SpringMVC执行流程 二、Vscode搭建SpringMVC框架 1、maven创建webapp原型项目 2、pom.xml下添加springmvc的相关依赖 3、在web.xml配置 4、springmvc.xml的配置 5、编写Controller控制器类 6、 编写JSP界面 7、项目结构图 一…

$subcribe的使用

$subcribe的使用 只要是store都有$subscribe函数&#xff0c;是订阅的意思&#xff0c;可以监测到store中数据的变化 使用$subscribe函数可以实现刷新不丢失&#xff0c;将数据保存到浏览器的本地存储中&#xff0c;每次进入页面都使用localStorage的数据填充页面

springboot带颜色的日志输出

话不多说直接贴全部代码在作解释 <?xml version"1.0" encoding"utf-8"?> <configuration><property name"pattern" value"%d{yyyy-MM-dd HH:mm:ss.SSS} [%thread] %-5level %logger{50} - %msg %n"/><propert…

MySQL数据库基础:使用、架构、SQL语句、存储引擎

文章目录 什么是数据库CS模式 基本使用安装链接服务器服务器、数据库、表关系简单使用数据库在Linux下的体现 MySQL架构连接器层客户端层服务层存储引擎层物理存储层 SQL分类存储引擎 什么是数据库 mysql&#xff1a;数据库服务的客户端mysqld&#xff1a;数据库服务的服务器端…

LVS精益价值管理系统 LVS.Web.ashx SQL注入漏洞复现

0x01 产品简介 LVS精益价值管理系统是杭州吉拉科技有限公司研发的一款专注于企业精益化管理和价值流优化的解决方案。该系统通过集成先进的数据分析工具、可视化的价值流映射技术和灵活的流程改善机制,帮助企业实现高效、低耗、高质量的生产和服务。 0x02 漏洞概述 LVS精益…