算法通关村第一关——链表经典问题之删除链表元素笔记

删除链表节点

总结一下高频常用的删除链表结点的情况,无论对链表进行何种操作,都需要精确查找精确指向 。另外,在删除链表节点时有一个很好用的技巧:虚头结点,将头结点的特殊性转化为一般,在后面具体阐述。

删除特定节点

常用方法 :直接模拟
关键 :精确找到特定节点,并且将其cur.next指向cur.next.next

由于可能删除头结点,可以引入一个虚头结点 dummyHead,来使得操作头结点的特殊性变得一般化,不用单独处理头结点,就和其它节点同等对待。

Code

java 复制代码
public static ListNode removeElements(ListNode head, int val) {
        ListNode dummyHead = new ListNode();
        //虚头结点的next指向头结点head
        dummyHead.next = head;
        ListNode cur = dummyHead;
        while (cur.next != null) {
            if (cur.next.val == val) {
            	//跳过一个节点指向,实现删除
                cur.next = temp.next.next;
            } else {
                cur = temp.next;
            }
        }
        //注意虚节点的next一定是真正的删除操作过后的链表头结点
        return dummyHead.next;
    }

删除倒数第K个节点

常用方法 :双指针
关键 :引入虚头结点 ,慢指针初始化在虚头节点,快指针初始化在原来的头节点,注意这里快节点是要走到null而不是它的下一个为null停止,因为这里快慢指针本来就有一位差距(倒也是好理解,如果都初始化在虚头节点则判定快节点的下一个为null)。

最后记得返回的是dummyHead.next

Code

java 复制代码
 public static ListNode removeNthFromEndByTwoPoints(ListNode head, int n) {
        ListNode dummyHead = new ListNode(0);
        dummyHead.next = head;
        ListNode fast= head;
        ListNode slow = dummyHead;
        while (n > 0){
        	fast = fast.next;
        	n--;
        }
        while (fast != null) {
            fast = fast.next;
            slow = slow.next;
        }
        slow.next = slow.next.next;
        return dummyHead.next;
    }

删除重复元素

重复元素保留一个

常用方法 :直接模拟
关键:真就直接模拟,没有任何套路,没有任何心机,就像Bob一样。

Code

java 复制代码
public ListNode deleteDuplicates(ListNode head) {
        if(head == null){
            return head;
        }
        ListNode cur = head;
        while(cur.next != null){
            if(cur.val == cur.next.val){
                cur.next = cur.next.next;
            }else{
             cur = cur.next;
            }
            
        }
        return head;
    }

重复元素全删

常用方法 :直接模拟
关键 :注意引入虚头节点并从虚头结点开始寻找,比较cur.next.valcur.next.next.val是否相等,如果相等,说明这个值是重复的,将其暂存

然后再往后找,并比较后面的节点值是否还为这个重复值。遇到了就将其删去(指向跳过)。

注意,由于一下往下看了两位节点,搜索要在它们都不为空的情况下进行。

Code

java 复制代码
public ListNode deleteDuplicates(ListNode head) {
        if (head == null) {
            return head;
        }
        ListNode dummy = new ListNode(-1111, head);
        ListNode cur = dummy;
        while (cur.next != null && cur.next.next != null) {
            if (cur.next.val == cur.next.next.val) {
                //关键在这个每一次都记录这个重复的数值
                int k = cur.next.val;
                //再消去所有val为这个重复数值的节点
                while (cur.next != null && cur.next.val == k) {
                    cur.next = cur.next.next;
                }
            } else {
                cur = cur.next;
            }
        }
        return dummy.next;
    }
相关推荐
StickToForever1 小时前
第4章 信息系统架构(五)
经验分享·笔记·学习·职场和发展
计算机小白一个3 小时前
蓝桥杯 Java B 组之设计 LRU 缓存
java·算法·蓝桥杯
万事可爱^3 小时前
HDBSCAN:密度自适应的层次聚类算法解析与实践
算法·机器学习·数据挖掘·聚类·hdbscan
敲敲敲-敲代码4 小时前
【SQL实验】触发器
数据库·笔记·sql
Moonnnn.5 小时前
51单片机学习——动态数码管显示
笔记·嵌入式硬件·学习·51单片机
大数据追光猿5 小时前
Python应用算法之贪心算法理解和实践
大数据·开发语言·人工智能·python·深度学习·算法·贪心算法
Dream it possible!6 小时前
LeetCode 热题 100_在排序数组中查找元素的第一个和最后一个位置(65_34_中等_C++)(二分查找)(一次二分查找+挨个搜索;两次二分查找)
c++·算法·leetcode
夏末秋也凉6 小时前
力扣-回溯-46 全排列
数据结构·算法·leetcode
南宫生6 小时前
力扣每日一题【算法学习day.132】
java·学习·算法·leetcode
柠石榴6 小时前
【练习】【回溯No.1】力扣 77. 组合
c++·算法·leetcode·回溯