力扣题目解析--两两交换链表中的节点

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

示例 1:

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

示例 2:

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

示例 3:

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

提示:

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

代码展示

cpp 复制代码
/**
 * Definition for singly-linked list.
 * struct ListNode {
 *     int val;
 *     ListNode *next;
 *     ListNode() : val(0), next(nullptr) {}
 *     ListNode(int x) : val(x), next(nullptr) {}
 *     ListNode(int x, ListNode *next) : val(x), next(next) {}
 * };
 */
class Solution {
public:
    ListNode* swapPairs(ListNode* head) {
        ListNode dummy(0);
        dummy.next=head;
        ListNode *prev=&dummy;

        while(prev->next!=nullptr&&prev->next->next!=nullptr){
            ListNode*first=prev->next;
            ListNode*second=prev->next->next;
            first->next=second->next;
            second->next=first;
            prev->next=second;
            prev=first;
        }
        return dummy.next;


    }
};

写者心得

这段代码有几个难点,首先就是那个虚拟头节点的设置,ListNode *prev=&dummy;这一步和往常的虚拟头节点设置是不一样的,因为我们后续需要把两个链表中的元素进行交换,而这一步是真正这个题目的核心一步,如果没有这一步,按着这个循环,就会把相邻的元素进行交换变成把链表颠倒过来,所以设置一个前置节点,再进行交换,就可以达成相邻元素进行交换。还有一个就是prev=first;其实都是为了达成相邻元素进行交换这个目的,这个就相当于是跳过目前已经交换的元素

代码的逐行解释

class Solution {
public:
    ListNode* swapPairs(ListNode* head) {
        // 创建一个虚拟头节点(dummy node),它的值为0。
        // 这个节点的存在是为了简化处理逻辑,特别是当原始链表的头节点也需要被交换时。
        ListNode dummy(0);
        
        // 让虚拟头节点的 next 指向原始链表的头节点。
        // 这样一来,即使原始链表的头节点发生变化,我们也可以通过 dummy.next 来访问新的头节点。
        dummy.next = head;
        
        // 创建一个指针 prev,它指向虚拟头节点。
        // prev 用于追踪当前要交换的两个节点对之前的节点。
        ListNode *prev = &dummy;

        // 使用 while 循环遍历链表,直到没有足够的节点可以进行交换。
        // 条件检查确保至少有两个节点可以交换:prev->next 和 prev->next->next 都不为空。
        while (prev->next != nullptr && prev->next->next != nullptr) {
            // 初始化 first 指针,指向当前要交换的第一个节点(即 prev 后面的第一个节点)。
            ListNode *first = prev->next;
            
            // 初始化 second 指针,指向当前要交换的第二个节点(即 first 的下一个节点)。
            ListNode *second = prev->next->next;

            // 让 first 指向原本 second 之后的节点。
            // 这一步是准备将 first 和 second 的位置交换,同时保证链表不会断裂。
            first->next = second->next;
            
            // 将 second 的 next 指向 first,完成这两个节点的位置交换。
            second->next = first;
            
            // 更新 prev 的 next 指针,使其指向新的头部节点(即 second),以反映交换后的链表结构。
            prev->next = second;
            
            // 移动 prev 指针到刚刚交换过的两个节点中的后一个节点(即原来的 first),以便为下一次迭代做准备。
            prev = first;
        }

        // 返回新的链表头节点,即虚拟头节点的 next 指针所指向的节点。
        return dummy.next;
    }
};
相关推荐
曲奇是块小饼干_40 分钟前
leetcode刷题记录(七十三)——543. 二叉树的直径
java·数据结构·算法·leetcode·职场和发展
Yuleave1 小时前
大型语言模型(LLM)在算法设计中的系统性综述
大数据·算法·语言模型
曲奇是块小饼干_1 小时前
leetcode刷题记录(七十二)——146. LRU 缓存
java·算法·leetcode·链表·职场和发展
tt5555555555551 小时前
每日一题-数组中的逆序对
数据结构·算法·排序算法
夏尔Gaesar6 小时前
pcm | Parity Check Matrix(奇偶校验矩阵)
算法·矩阵·pcm
m0_dawn9 小时前
《贪心算法:原理剖析与典型例题精解》
python·算法·职场和发展·贪心算法·蓝桥杯
invincible_Tang9 小时前
贪心算法(题2)最大不相交区间数量
算法·贪心算法
AIzealot无11 小时前
力扣hot100之螺旋矩阵
算法·leetcode·矩阵
兑生11 小时前
力扣面试150 长度最小的子数组 滑动窗口
算法·leetcode·面试
miilue11 小时前
[LeetCode] 链表I — 704#设计链表 | 203#移除链表元素 | 206#反转链表 | 递归法
java·开发语言·c++·算法·leetcode·链表