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

博客围绕LeetCode中两两交换链表相邻节点的题目展开。要求不修改节点内部值,仅进行节点交换。解题思路是用两个指针指向需交换的节点,找到循环规律,同时要单独处理最初两个节点的交换,并根据节点数奇偶性添加指针轮空判定。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

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

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

题解

class Solution:
    def swapPairs(self, head: Optional[ListNode]) -> Optional[ListNode]:
        p = head
        if not p: # 没有节点
            return p
        q = p.next
        if not q: # 只有一个节点
            return p
        # 最初两个节点的交换
        p.next = q.next
        q.next = p
        head = q # 找对头,交换后,原来的头变成第二了
        # 后续节点交换
        while q:
            l = p
            p = p.next
            if not p: # 偶数个节点,p先轮空
                return head
            q = p.next
            if not q: # 奇数个节点,q先轮空
                return head
            l.next = q
            p.next = q.next
            q.next = p
        return head # 多余的

思路

找规律。使用两个指针指向需要交换的node,示例1中,p指向1,q指向2,令他们交换位置;交换玩后,变成q在前p在后;这时,令p向后移动一个,q继续指向p的后一个node;思考到这里就找到循环的规律了。
接着就是细节,最初两个节点的交换和后面不一样,单独写出来;当节点数为偶数时,会先出现p指针轮空,单节点数为奇数时,是q指针先轮空,所以合适的地方加上判定。

### 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
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值