「日更刷题」24. 两两交换链表中的节点

简介: 「日更刷题」24. 两两交换链表中的节点

一、前言

每日算法坚持第二天,该专栏争取日日更新,加油

二、简介和示例

简介

给你一个链表,两两交换其中相邻的节点,并返回交换后链表的头节点。你必须在不修改节点内部的值的情况下完成本题(即,只能进行节点交换)。

链表数据结构

public class ListNode {
    public int val;
    public ListNode next;
    public ListNode() {
    }
    public ListNode(int val) {
        this.val = val;
    }
    public ListNode(int val, ListNode next) {
        this.val = val;
        this.next = next;
    }
}
复制代码

示例1

网络异常,图片无法展示
|

输入: head = [1,2,3,4]
输出: [2,1,4,3]
复制代码

示例2

输入: head = []
输出: []
复制代码

示例3

输入: head = [1]
输出: [1]
复制代码

提示

  • 链表中节点的数目在范围 [0, 100]
  • 0 <= Node.val <= 100

三、解题

思路分析

这个题的思路有两种, 一种是使用递归来做, 一种是使用虚拟头节点来做, 我使用的是虚拟头节点方法

首先,我们假设链表结构如图所示

网络异常,图片无法展示
|

我们需要两两交换链表中的节点,变成如图所示的结构

网络异常,图片无法展示
|

这边我们使用虚拟头节点的方式,为该链表创建一个头节点,不然的话没有指针指向第一个节点,不好去操作

网络异常,图片无法展示
|

我们要进行的操作如下:

  • 将节点2放置到节点1的位置
  • 节点1的 next指向后面不变的节点3地址
  • 节点2的 next指向节点1

网络异常,图片无法展示
|

代码实现

方法入参是 ListNode head

首先我们新建一个虚拟节点, 并将其指向我们的链表 head, 同时返回的时候注意不要直接返回

ListNode dummyNode = new ListNode(0);
dummNode.next = head;
...
return dummNode.next;
复制代码

设置一个我们要对其进行修改的链表

ListNode prev = dumm
复制代码

接下来就要对我们的链表进行循环操作,去对节点进行位置交换

但是这里有一个地方需要注意,那就是循环的中止条件,如下图所示,进行第三次循环的时候链表只剩下一位,不能够进行两两交换了,那肯定要中止循环了,这个时候的判断条件是 链表.next.next != null, 但是这是基于我们的链表节点是单数的情况下, 如果是偶数的情况下, 我们循环的终止条件应该是 链表.next != null, 而且这两个条件的执行顺序也要注意,应该是 链表.next != null 在前,因为 链表为null 的情况下 链表.next.next会报空指针异常

网络异常,图片无法展示
|

具体代码如下:

while(prev.next !=null && prev.next.next != null){
}
复制代码

在循环内具体的操作:

  • 缓存后面不变的节点3
  • 缓存1节点
  • prev.next更改为2节点
  • 1节点.next = 3节点
  • 2节点.next = 1节点
  • 将prev跟改为还未交换位置的节点
  • 具体代码如下所示
// 缓存不变的链表信息(只动两个节点进行位置交换, 将两个节点之后的节点缓存下来)
ListNode temp = prev.next.next.next;
// 缓存第一个节点信息(先将第二个节点的位置放到第一个节点的位置上, 第一个节点就没有了节点信息)
ListNode temp1 = prev.next;
// 将第二个节点提取到第一个节点的位置
prev.next = prev.next.next;
// 将不变的链表放置到 原第一个节点后面
temp1.next = temp;
// 将原第一个节点放置到现第二个节点的位置
prev.next.next = temp1;
// 链表移动两位
prev = prev.next.next;
复制代码

四、代码展示

public static ListNode swapPairs(ListNode head) {
    // 设置虚拟头节点
    ListNode dummyNode = new ListNode(0);
    dummyNode.next = head;
    ListNode prev = dummyNode;
    // 先进行下一个节点的判断, 在进行下下个节点的判断, 避免空指针异常
    while (prev.next != null && prev.next.next != null) {
        // 缓存不变的链表信息(只动两个节点进行位置交换, 将两个节点之后的节点缓存下来)
        ListNode temp = prev.next.next.next;
        // 缓存第一个节点信息(先将第二个节点的位置放到第一个节点的位置上, 第一个节点就没有了节点信息)
        ListNode temp1 = prev.next;
        // 将第二个节点提取到第一个节点的位置
        prev.next = prev.next.next;
        // 将不变的链表放置到 原第一个节点后面
        temp1.next = temp;
        // 将原第一个节点放置到现第二个节点的位置
        prev.next.next = temp1;
        // 链表移动两位
        prev = prev.next.next;
    }
    return dummyNode.next;
}
复制代码

五、提交代码

网络异常,图片无法展示
|



目录
相关文章
|
3月前
LeetCode第二十四题(两两交换链表中的节点)
这篇文章介绍了LeetCode第24题的解法,即如何通过使用三个指针(preNode, curNode, curNextNode)来两两交换链表中的节点,并提供了详细的代码实现。
33 0
LeetCode第二十四题(两两交换链表中的节点)
|
3月前
Leetcode第十九题(删除链表的倒数第N个节点)
LeetCode第19题要求删除链表的倒数第N个节点,可以通过快慢指针法在一次遍历中实现。
49 0
Leetcode第十九题(删除链表的倒数第N个节点)
05_删除链表的倒数第N个节点
05_删除链表的倒数第N个节点
|
3月前
(剑指offer)18、删除链表的节点—22、链表中倒数第K个节点—25、合并两个排序的链表—52、两个链表的第一个公共节点(2021.12.07)
(剑指offer)18、删除链表的节点—22、链表中倒数第K个节点—25、合并两个排序的链表—52、两个链表的第一个公共节点(2021.12.07)
58 0
04_两两交换链表中的节点
04_两两交换链表中的节点
|
8月前
【移除链表元素】LeetCode第203题讲解
【移除链表元素】LeetCode第203题讲解
|
7月前
|
存储 SQL 算法
LeetCode力扣第114题:多种算法实现 将二叉树展开为链表
LeetCode力扣第114题:多种算法实现 将二叉树展开为链表
|
7月前
|
存储 SQL 算法
LeetCode 题目 86:分隔链表
LeetCode 题目 86:分隔链表
|
7月前
|
存储 算法 Java
【经典算法】Leetcode 141. 环形链表(Java/C/Python3实现含注释说明,Easy)
【经典算法】Leetcode 141. 环形链表(Java/C/Python3实现含注释说明,Easy)
68 2
|
8月前
<数据结构>五道LeetCode链表题分析.环形链表,反转链表,合并链表,找中间节点.
<数据结构>五道LeetCode链表题分析.环形链表,反转链表,合并链表,找中间节点
80 1