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

给定一个链表,两两交换其中相邻的节点,并返回交换后的链表。
你不能只是单纯的改变节点内部的值,而是需要实际的进行节点交换。

提示:
链表中节点的数目在范围 [0, 100] 内
0 <= Node.val <= 100
两两交换链表中的节点

e.g. 1->2->3->4 =>2->1->4->3
1->2->3->4->5 =>2->1->4->3->5
就是从第二个节点开始,每个节点的的next都往前面指;
而head怎么办呢?
特判:最后的head是原第二个节点;

class Solution {
public:
    ListNode* swapPairs(ListNode* head) {
        if(head==NULL || head->next==NULL)//链表为空、有一个节点 
            return head; 
        ListNode *pre=NULL,*tail=NULL,*now=NULL;
        pre=NULL;
        now=head;
        while(now!=NULL){
            tail=now->next;
            if(now==head){
                now->next=tail->next;
                tail->next=now;
                head=tail;
            }else{
                if(tail==NULL)
					break;//奇数 ,此时的now是链表的最后一个 
				 else{
				 	pre->next=tail;
				 	now->next=tail->next;
				 	tail->next=now;
				 }
            }
			pre=now;
			now=now->next;
        }
        return head;
    }
};
### 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、付费专栏及课程。

余额充值