小阳同学刷题日记-203. 移除链表元素

        题目: 给你一个链表的头节点 head 和一个整数 val ,请你删除链表中所有满足 Node.val == val 的节点,并返回 新的头节点 。

        思路: 

        两种思路:① 不创建虚拟节点的思路;② 创建虚拟节点的思路

① 不创建虚拟节点的思路

  1. 首先处理头节点,如果头节点的值等于 val,则将头节点后移直到其值不等于 val
  2. 然后遍历链表,删除所有值为 val 的节点。
  3. 需要维护一个指针 prev 指向当前节点的前一个节点,以便在删除节点时重新连接链表。
  4. 删除节点的方法是将 prev 节点的 next 指针指向当前节点的下一个节点,并释放当前节点的内存空间。
  5. 如果当前节点的值不等于 val,则将 prev 指针移动到当前节点,继续遍历下一个节点。
  6. 遍历结束后,返回头节点作为新的头节点

 ② 创建虚拟节点的思路

  1. 创建一个虚拟头节点 dummyHead,使其 next 指向原链表的头节点 head
  2. 使用指针 cur 遍历链表,初始时指向虚拟头节点 dummyHead
  3. 在遍历过程中,如果当前节点的下一个节点的值等于 val,则删除该节点。
  4. 删除节点的方法是将当前节点的 next 指针指向下下个节点,并释放被删除节点的内存空间。
  5. 如果当前节点的下一个节点的值不等于 val,则将 cur 指针移动到下一个节点。
  6. 遍历结束后,返回虚拟头节点 dummyHead 的下一个节点作为新的头节点。

         代码:

① 不创建虚拟节点的代码 

class Solution {
public:
    ListNode* removeElements(ListNode* head, int val) {
        // 删除头节点
        while(head != NULL && head->val == val){ // 当头节点的值等于给定值时循环删除
            ListNode* temp = head; // 临时指针指向头节点
            head = head->next; // 移动头指针到下一个节点
            delete temp; // 释放被删除的头节点
        }

        // 删除非头节点
        ListNode* cur = head; // 使用cur指针遍历链表,初始指向头节点
        while(cur != NULL && cur->next != NULL){ // 循环直到cur指针指向链表尾节点的下一个节点或cur指针为空
            if(cur->next->val == val){ // 如果当前节点的下一个节点的值等于给定值
                ListNode* temp = cur->next; // 临时指针指向当前节点的下一个节点
                cur->next = cur->next->next; // 将当前节点的下一个节点指针指向下下个节点
                delete temp; // 释放被删除的节点
            }
            else{ // 如果当前节点的下一个节点的值不等于给定值
                cur = cur->next; // 移动当前节点指针到下一个节点
            }
        }

        return head; // 返回处理后的链表头节点指针
    }
};

 ② 创建虚拟节点的代码

class Solution {
public:
    ListNode* removeElements(ListNode* head, int val) {
        ListNode* dummyHead = new ListNode(0); // 创建虚拟头节点,值为0
        dummyHead->next = head; // 将虚拟头节点指向原链表的头节点
        ListNode* cur = dummyHead; // 使用cur指针遍历链表,初始指向虚拟头节点
        while(cur->next != NULL){ // 循环直到cur指针指向链表尾节点的下一个节点或cur指针为空
            if(cur->next->val == val){ // 如果当前节点的下一个节点的值等于给定值
                ListNode* temp = cur->next; // 临时指针指向当前节点的下一个节点
                cur->next = cur->next->next; // 将当前节点的下一个节点指针指向下下个节点
                delete temp; // 释放被删除的节点
            }
            else{ // 如果当前节点的下一个节点的值不等于给定值
                cur = cur->next; // 移动当前节点指针到下一个节点
            }            
        }

        return dummyHead->next; // 返回处理后的链表头节点指针,即虚拟头节点的下一个节点
    }
};

两种方法各有优缺点,我们来分析一下:

方法一:使用虚拟头节点

优点:

  1. 简化了对头节点的处理:由于引入了虚拟头节点,不需要特殊处理头节点,统一了对节点的处理逻辑。
  2. 避免了对空链表的特殊情况处理:即使链表为空,也可以直接操作虚拟头节点,无需额外的判断和处理。

缺点:

  1. 需要额外的空间:引入了虚拟头节点,会占用额外的内存空间。
  2. 删除节点时需要注意细节:删除节点时,需要特别注意处理头节点的情况,以及确保链表的连接性。

方法二:不使用虚拟头节点

优点:

  1. 不需要额外的空间:不引入虚拟头节点,节省了额外的内存空间。
  2. 逻辑较为清晰:对头节点的处理和其他节点的处理分开,逻辑相对清晰。

缺点:

  1. 需要特殊处理头节点:由于没有虚拟头节点,需要单独处理头节点的情况,会增加代码的复杂度。
  2. 需要处理空链表的特殊情况:在链表为空时,需要额外的判断和处理。

        综上所述,两种方法各有优缺点,具体选择哪种方法取决于实际情况和个人偏好。如果对空间复杂度要求较高,且能够确保代码逻辑正确性,可以选择不使用虚拟头节点的方法;如果追求代码的简洁和清晰,且能够接受额外的空间消耗,可以选择使用虚拟头节点的方法。

 

努力学习,天天向上!

  • 25
    点赞
  • 17
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值