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

题目

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

示例 1:
在这里插入图片描述
输入:head = [1,2,3,4]
输出:[2,1,4,3]
示例 2:

输入:head = []
输出:[]
示例 3:

输入:head = [1]
输出:[1]

提示:

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

代码

/**
 * 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=new ListNode(0,head);//创建一个节点,数值为0,指向的下一个节点为头结点,相当于在原链表的头部插入一个新节点
        ListNode* pre=dummy;
        ListNode* first=head;
        ListNode* second=NULL;//因为不知道刚开始头结点后是否有数据
        while(first&&first->next)
        {
            second=first->next;
            ListNode* nxt=second->next;//因为nxt在随时变化
            pre->next=second;//因为从这一步开始到倒数第三步是断链重连
            second->next=first;
            first->next=nxt;
            pre=first;//后两步是在断链重连后,更新指针指向为下一轮交换做准备
            first=nxt;
        }
        return dummy->next;//dummy指向并没有改变,直接返回dummy的下一个节点即链表头结点
    }
};

原理图

在这里插入图片描述

原理解释

提示:算法流程及解释在代码中已标注
增加一个虚拟节点为了避免掉很多边界条件的处理

需要定义四个指针:
pre指针:指向需要交换节点的前一个节点
first指针:指向第一个需要交换的节点
second指针:指向要交换的第二个节点
nxt指针:指向下一组要交换节点的第一个节点

流程:
①second的下一个节点指向first
②first指针的下一个节点指向nxt
③pre指针的下一个节点指向second
④pre后移一个节点指向first
⑤first指向的节点更改为下一轮需要交换节点的第一个节点nxt

小结

这个算法可以应用在各种需要交换节点的场景,如反转链表、两两交换节点等操作。在实际的面试或工程中,对链表操作的熟练掌握能够帮助我们更好地解决问题。

在这里插入图片描述
原理图借鉴哔站华南溜达虎,如有侵权联系删除。

### C语言实现链表节点两两交换 为了在C语言中实现链表节点两两交换,可以采用迭代的方法。此方法通过引入一个虚拟头节点简化边界条件处理,并利用三个指针完成节点交换过程。 #### 定义链表结构体 首先定义单项链表数据结构: ```c typedef struct ListNode { int val; struct ListNode *next; } ListNode; ``` #### 创建辅助函数用于创建新节点 这有助于测试和验证最终算法的功能: ```c ListNode* createNode(int value) { ListNode* newNode = (ListNode*)malloc(sizeof(ListNode)); newNode->val = value; newNode->next = NULL; return newNode; } ``` #### 主要逻辑:两两交换链中的节点 核心思路在于维护前驱节点`prev`、当前节点`first`以及后续节点`second`之间的关系,在遍历过程中不断调整这些指针的位置以达到交换目的[^3]。 ```c void swapPairs(ListNode **headRef) { // 如果列为空或只有一个元素,则无需任何操作直接返回原列 if (*headRef == NULL || (*headRef)->next == NULL) { return; } // 创建一个新的哨兵节点作为新的头部之前的一个位置 ListNode *dummyHead = createNode(0); dummyHead->next = *headRef; // `prev`始终指向待交换的一对节点之前的那个节点 ListNode *prev = dummyHead; while ((*headRef != NULL) && ((*headRef)->next != NULL)) { // 记录下一对需要处理的第一个节点 ListNode *first = *headRef; // 更新 head 指向第二节点之后的部分 *headRef = first->next->next; // 进行实际的节点交换 prev->next = first->next; // 将 prev 的 next 设置为 second first->next->next = first; // 把原来的第二个节点连接回第一个节点后面 first->next = *headRef; // 原来的第一个节点现在应该指向剩余部分 // 移动 prev 到刚刚被交换过的最后一个节点处准备下次循环 prev = first; } // 返回修改后的链表的新起点 *headRef = dummyHead->next; free(dummyHead); // 清理临时使用的哨兵节点 } ``` 上述代码实现了完整的两两交换功能,其中特别注意到了对于奇数长度链表最后剩下单独一个节点的情况也能够正确处理[^2]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

小辉同志

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值