二、链表(2):移除链表元素

news/2025/2/22 15:11:09

https://leetcode-cn.com/problems/remove-linked-list-elements/

题意:删除链表中等于给定值 val 的所有节点。

示例 1:
输入:head = [1,2,6,3,4,5,6], val = 6
输出:[1,2,3,4,5]

示例 2:
输入:head = [], val = 1
输出:[]

示例 3:
输入:head = [7,7,7,7], val = 7
输出:[]

一、思路

这里以链表 1 4 2 4 来举例,移除元素4。

203_<a class=链表删除元素1" src="https://img-blog.csdnimg.cn/20210316095351161.png" />

如果使用C,C++编程语言的话,不要忘了还要从内存中删除这两个移除的节点, 清理节点内存之后如图:

203_<a class=链表删除元素2" src="https://img-blog.csdnimg.cn/20210316095418280.png" />

 当然如果使用java ,python的话就不用手动管理内存了。

还要说明一下,就算使用C++来做leetcode,如果移除一个节点之后,没有手动在内存中删除这个节点,leetcode依然也是可以通过的,只不过,内存使用的空间大一些而已,但建议依然要养生手动清理内存的习惯。

这种情况下的移除操作,就是让节点next指针直接指向下下一个节点就可以了,那么因为单链表的特殊性,只能指向下一个节点,刚刚删除的是链表的中第二个,和第四个节点,那么如果删除的是头结点又该怎么办呢?

这里就涉及如下链表操作的两种方式:

  • 直接使用原来的链表来进行删除操作。
  • 设置一个虚拟头结点在进行删除操作。
  • 来看第一种操作:直接使用原来的链表来进行移除

203_<a class=链表删除元素3" src="https://img-blog.csdnimg.cn/2021031609544922.png" />

 移除头结点和移除其他节点的操作是不一样的,因为链表的其他节点都是通过前一个节点来移除当前节点,而头结点没有前一个节点。所以头结点如何移除呢,其实只要将头结点向后移动一位就可以,这样就从链表中移除了一个头结点

203_<a class=链表删除元素4" src="https://img-blog.csdnimg.cn/20210316095512470.png" />

依然别忘将原头结点从内存中删掉。

 203_<a class=链表删除元素5" src="https://img-blog.csdnimg.cn/20210316095543775.png" />

 这样移除了一个头结点,是不是发现,在单链表中移除头结点 和 移除其他节点的操作方式是不一样,其实在写代码的时候也会发现,需要单独写一段逻辑来处理移除头结点的情况。那么可不可以 以一种统一的逻辑来移除 链表的节点呢。其实可以设置一个虚拟头结点,这样原链表的所有节点就都可以按照统一的方式进行移除了。来看看如何设置一个虚拟头。依然还是在这个链表中,移除元素1。

203_<a class=链表删除元素6" src="https://img-blog.csdnimg.cn/20210316095619221.png" />

 这里来给链表添加一个虚拟头结点为新的头结点,此时要移除这个旧头结点元素1。这样是不是就可以使用和移除链表其他节点的方式统一了呢?来看一下,如何移除元素1 呢,还是熟悉的方式,然后从内存中删除元素1。最后呢在题目中,return 头结点的时候,别忘了 return dummyNode->next;, 这才是新的头结点

二、C++代码

  • 直接使用原来的链表来进行移除节点操作:
class Solution {
public:
    ListNode* removeElements(ListNode* head, int val) {
        // 删除头结点
        while (head != NULL && head->val == val) { // 注意这里不是if
            ListNode* tmp = head;
            head = head->next;
            delete tmp;
        }

        // 删除非头结点
        ListNode* cur = head;
        while (cur != NULL && cur->next!= NULL) {
            if (cur->next->val == val) {
                ListNode* tmp = cur->next;
                cur->next = cur->next->next;
                delete tmp;
            } else {
                cur = cur->next;
            }
        }
        return head;
    }
};



-----------------------------------------------------------------------------------------


#include <iostream>
#include<cstdlib>

using namespace std;



// 单链表
struct ListNode {
    int val;  // 节点上存储的元素
    ListNode *next;  // 指向下一个节点的指针
    ListNode(int x) : val(x), next(NULL) {}  // 节点的构造函数
};


 
ListNode* RemoveElement(ListNode* head, int val) {

   // 移除head;
   while (head != NULL && head->val == val) {
              ListNode* tem = head;
              head = head->next;
              delete tem;
          }
          
   // 删除非头节点;
   ListNode* cur = head;
   while (cur != NULL && cur->next != NULL) {
      
      if (cur->next->val == val) {
          ListNode* tem = cur->next;
          cur->next == cur->next->next;
          delete tem;
      }
      else
      cur == cur->next;
  }        
          
    return head;      
}
    
 


int main() {
   cout << "Hello World" << endl; 
   
   return 0;
}






  • 设置一个虚拟头结点在进行移除节点操作:
class Solution {
public:
    ListNode* removeElements(ListNode* head, int val) {
        ListNode* dummyHead = new ListNode(0); // 设置一个虚拟头结点
        dummyHead->next = head; // 将虚拟头结点指向head,这样方面后面做删除操作
        ListNode* cur = dummyHead;
        while (cur->next != NULL) {
            if(cur->next->val == val) {
                ListNode* tmp = cur->next;
                cur->next = cur->next->next;
                delete tmp;
            } else {
                cur = cur->next;
            }
        }
        head = dummyHead->next;
        delete dummyHead;
        return head;
    }
};
  • 作者微信:程序员Carl(opens new window)
  • B站视频:代码随想录(opens new window)
  • 知识星球:代码随想录

http://www.niftyadmin.cn/n/884097.html

相关文章

faster rcnn 损失函数

损失函数分为框预测回归和分类loss&#xff0c;每个都分为rpn和rcnn。 rpn损失&#xff1a; _anchor_target_layer 给图像内部的anchors分配rpn_labels(1或0&#xff0c;前景或背景)&#xff0c;再和rpn_cls_score做交叉熵&#xff0c;此时只关注物体是前景/背景&#xff0c;…

二、链表(3):设计链表

https://leetcode-cn.com/problems/design-linked-list/ 题意&#xff1a; 在链表类中实现这些功能&#xff1a; get(index)&#xff1a;获取链表中第 index 个节点的值。如果索引无效&#xff0c;则返回-1。addAtHead(val)&#xff1a;在链表的第一个元素之前添加一个值为 val…

faster rcnn 图像识别三大步骤

三大步骤分别是识别、分割、分类。 识别&#xff1a;识别是为了得到rois(region of interests&#xff0c;即感兴趣的区域)。首先在卷基层提取特征&#xff0c;再在每个特征上生成anchor[1]&#xff0c;把anchor在_region_proposal [2]做识别&#xff0c;提取部分anchor得到ro…

二、链表(4):反转链表

https://leetcode-cn.com/problems/reverse-linked-list/ 题意&#xff1a;反转一个单链表。 示例: 输入: 1->2->3->4->5->NULL输出: 5->4->3->2->1->NULL一、思路 如果再定义一个新的链表&#xff0c;实现链表元素的反转&#xff0c;其实这是对…

机器学习和深度学习中,L2正则化为什么能防止过拟合?

正则化是为了降低模型的复杂度&#xff0c;模型过于复杂&#xff0c;则过拟合&#xff1b; 与傅里叶变换类似&#xff0c;高频的部分表示细节&#xff0c;尽量减少高频部分的影响&#xff1b; 傅里叶级数也是&#xff0c;高阶表示细节&#xff1b; 当阶数较高时&#xff0c;…

算法——找出数组中出现次数最多的数

今天面试中有一道算法题&#xff0c;好长时间没做算法了&#xff0c;虽然很简单&#xff0c;但是当时没做出来&#xff0c;记录一下 import numpy as npa np.array([1,4,5,6,4,7,5,4])#找出数组中出现次数最多的数 def find_a(arr):arr_sort np.sort(arr)ll np.size(arr)s …

句子中的单词反转

import numpy as npstr This is a cat length len(str) reset s length arr np.arange(length)for i in arr[::-1]:if str[i] :reset reset str[i:s]s ireset reset str[:s] print(reset)输出&#xff1a; cat a is This

二、链表(5):两两交换链表中的节点

https://leetcode-cn.com/problems/swap-nodes-in-pairs/ 给定一个链表&#xff0c;两两交换其中相邻的节点&#xff0c;并返回交换后的链表。你不能只是单纯的改变节点内部的值&#xff0c;而是需要实际的进行节点交换。一、思路 这道题目正常模拟就可以了。 建议使用虚拟头…