代码随想录算法训练营第03天 | LeetCode 203.移除链表元素,707.设计链表,206.反转链表

这篇博客主要介绍了LeetCode中的链表题目,包括203题移除链表元素,707题设计链表,以及206题反转链表。讲解了如何在链表中删除特定值的节点,如何设计一个链表类以实现常见操作,以及如何反转链表。此外,还讨论了链表与数组的区别以及链表的基础概念。
摘要由CSDN通过智能技术生成

LeetCode [203. 移除链表元素]

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

  • 示例 1:removelinked-list

    输入: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. 直接使用原来的链表来进行删除操作

  2. 添加一个虚拟头结点:

    在单链表中移除头结点和移除其他节点的操作方式是不一样,需要单独写一段逻辑来处理移除头结点的情况。
    通过设置一个虚拟头结点,这样原链表的所有节点就都可以按照统一的方式进行移除了。

  3. 递归:

    链表的定义具有递归的性质,因此链表题目常可以用递归的方法求解。

    对于给定的链表,首先对除了头节点head 以外的节点进行删除操作,然后判断head 的节点值是否等于给定的 val。

    如果head 的节点值等于 val,则head 需要被删除,因此删除操作后的头节点为head.next;如果head 的节点值不等于 val,则head 保留,因此删除操作后的头节点还是head。上述过程是一个递归的过程。

    递归的终止条件是head 为空,此时直接返回head。当head 不为空时,递归地进行删除操作,然后判断head 的节点值是否等于 val 并决定是否要删除head。

//直接使用原来的链表来进行删除操作
/**
 * Definition for singly-linked list.
 * public class ListNode {
 *     int val;
 *     ListNode next;
 *     ListNode() {}
 *     ListNode(int val) { this.val = val; }
 *     ListNode(int val, ListNode next) { this.val = val; this.next = next; }
 * }
 */
class Solution {
    public ListNode removeElements(ListNode head, int val) {
        
        /*
         *删除头结点,头结点存在且头结点的值等于val时
         *将新的头结点设置为head.next(头结点向后移动一位)并删除原来的头结点(Java自带回收机制,不需要手动清除)
         */
        while(head != null && head.val == val){
            head = head.next;
        }
        /*
         *head已经为null,提前退出,返回结果
         */
        if(head == null){
            return head;
        }
        /*
         *删除非头节点
         *在"下个节点1"存在的情况下
         *如果"下个节点1"的值等于目标值,删除"下个节点1"(方式是当前节点的next指向"下个节点1"的"下个节点2")
         *如果"下个节点1"的值不等于目标值,则将"当前节点"移动到"下个节点1",保留"下个节点1"
         *将pre和cur移动到下一个节点
         *"当前节点"的下一个节点为空时,链表遍历结束,此时所有节点值等于val的节点都被删除。
         */
        ListNode pre = head;
        ListNode cur = head.next;
        while(cur != null){
            if(cur.val == val){
                pre.next = cur.next;
            }else{
                pre = cur;
            }
            cur = cur.next;
        }
        return head;
    }
}
//添加一个虚拟头结点法
class Solution {
    public ListNode removeElements(ListNode head, int val) {
        if(head == null){
            return head;
        }
        ListNode dummyNode = new ListNode(-1,head);
        ListNode pre = dummyNode;
        ListNode cur = head;
        while(cur != null){
            if(cur.val == val){
                pre.next = cur.next;
            }else{
                pre = cur;
            }
            cur = cur.next;
        }
        return dummyNode.next;
    }
}
//递归
class Solution {
    public ListNode removeElements(ListNode head, int val) {
        if (head == null) {
            return head;
        }
        head.next = removeElements(head.next, val);
        return head.val == val ? head.next : head;
    }
}

扩展:链表和数组的区别;链表的定义

  • 链表和数组的区别
  1. 数组是在内存中是连续分布的,但是链表在内存中可不是连续分布的。链表是通过指针域的指针链接在内存中各个节点。
  2. 数组在定义的时候,长度就是固定的,如果想改动数组的长度,就需要重新定义一个新的数组。
  3. 链表的长度可以是不固定的,并且可以动态增删, 适合数据量不固定,频繁增删,较少查询的场景。
  • 链表的定义
public class ListNode {
    // 结点的值
    int val;

    // 下一个结点
    ListNode next;

    // 节点的构造函数(无参)
    public ListNode() {
    }

    // 节点的构造函数(有一个参数)
    public ListNode(int val) {
        this.val = val;
    }

    // 节点的构造函数(有两个参数)
    public ListNode(int val, ListNode next) {
        this.val = val;
        this.next = next;
    }
}

LeetCode [707. 设计链表]

题目:设计链表的实现。您可以选择使用单链表或双链表。单链表中的节点应该具有两个属性:val 和 next。val 是当前节点的值,next 是指向下一个节点的指针/引用。如果要使用双向链表,则还需要一个属性 prev 以指示链表中的上一个节点。假设链表中的所有节点都是 0-index 的。

在链表类中实现这些功能:

get(index):获取链表中第 index 个节点的值。如果索引无效,则返回-1。
addAtHead(val):在链表的第一个元素之前添加一个值为 val 的节点。插入后,新节点将成为链表的第一个节点。
addAtTail(val):将值为 val 的节点追加到链表的最后一个元素。
addAtIndex(index,val):在链表中的第 index 个节点之前添加值为 val 的节点。如果 index 等于链表的长度,则该节点将附加到链表的末尾。如果 index 大于链表长度,则不会插入节点。如果index小于0,则在头部插入节点。
deleteAtIndex(index):如果索引 index 有效,则删除链表中的第 index 个节点。

  • 示例 1:

    MyLinkedList linkedList = new MyLinkedList();
    linkedList.addAtHead(1);
    linkedList.addAtTail(3);
    linkedList.addAtIndex(1,2); //链表变为1-> 2-> 3
    linkedList.get(1); //返回2
    linkedList.deleteAtIndex(1); //现在链表是1-> 3
    linkedList.get(1); //返回3

思路:基础不牢,地动山摇,这个题目涵盖了链表很多的基本操作。直接选择看题解。

//单链表
class ListNode {
    int val;
    ListNode next;
    ListNode(){}
    ListNode(int val) {
        this.val=val;
    }
    ListNode(int val,ListNode next){
        this.val=val;
        this.next=next;
    }
}
class MyLinkedList {
    //size存储链表元素的个数
    int size;
    //设置虚拟头结点
    ListNode dummyNode;
    //初始化链表
    public MyLinkedList(){
        size = 0;
        dummyNode = new ListNode(0);
    }
    //获取第n个节点的数值
    public int get(int index) {
        //如果index非法,则返回-1
        if(index < 0 || index >= size){
            return -1;
        }
        ListNode cur = dummyNode;
        //因为第一个是虚拟头结点,所以查询的时候从index + 1开始
        for(int i = 0;i <= index;i++){
            cur = cur.next;
        }
        return cur.val;
    }
    //在链表头部前插入一个节点
    public void addAtHead(int val) {
        addAtIndex(0,val);
    }
    //在链表尾部后插入一个节点
    public void addAtTail(int val) {
        addAtIndex(size,val);
    }
    // 在第 index 个节点之前插入一个新节点,例如index为0,那么新插入的节点为链表的新头节点。
    // 如果 index 等于链表的长度,则说明是新插入的节点为链表的尾结点
    // 如果 index 大于链表的长度,则返回空
    public void addAtIndex(int index, int val) {
        if(index > size){
            return;
        }
        if(index < 0){
            index = 0;
        }
        size++;
        //找到需要插入节点的前驱
        ListNode pre = dummyNode;
        for(int i = 0;i < index;i++){
            pre = pre.next;
        }
        ListNode add = new ListNode(val);
        add.next = pre.next;
        pre.next = add;
    }
    //删除第index个节点
    public void deleteAtIndex(int index) {
        if(index < 0 || index >= size){
            return;
        }
        size--;
        if(index == 0){
            dummyNode = dummyNode.next;
            return;
        }
        ListNode pre = dummyNode;
        for(int i = 0;i < index;i++){
            pre = pre.next;
        }
        pre.next = pre.next.next;
    }
}

LeetCode [206. 反转链表]

题目:给你单链表的头节点 head ,请你反转链表,并返回反转后的链表。

  • 示例 1:

    输入:head = [1,2,3,4,5]
    输出:[5,4,3,2,1]

  • 示例 2:

    输入:head = [1,2]
    输出:[2,1]

  • 示例 3:

    输入:head = []
    输出:[]

思路

//双指针法
/**
 * Definition for singly-linked list.
 * public class ListNode {
 *     int val;
 *     ListNode next;
 *     ListNode() {}
 *     ListNode(int val) { this.val = val; }
 *     ListNode(int val, ListNode next) { this.val = val; this.next = next; }
 * }
 */
class Solution {
    public ListNode reverseList(ListNode head) {
        ListNode pre = null;
        ListNode cur = head;
        //next用来保存cur的下一个节点
        ListNode next = null;
        //不断遍cur
        //每次迭代到 cur,都将 cur 的 next 指向 pre,然后 pre 和 cur 前进一位。
        //都迭代完了(cur 变成 null 了),pre 就是最后一个节点了。
        while(cur != null){
            next = cur.next;
            //把当前节点的next指向另一个方向
            cur.next = pre;
            //pre 和 cur 前进一位。
            pre = cur;
            cur = next;
        }
        return pre;
    }
}
//递归
class Solution {
	public ListNode reverseList(ListNode head) {
		//递归终止条件是当前为空,或者下一个节点为空
		if(head==null || head.next==null) {
			return head;
		}
		//这里的cur就是最后一个节点
		ListNode cur = reverseList(head.next);
		//如果链表是 1->2->3->4->5,那么此时的cur就是5
		//而head是4,head的下一个是5,下下一个是空
		//所以head.next.next 就是5->4
		head.next.next = head;
		//防止链表循环,需要将head.next设置为空
		head.next = null;
		//每层递归函数都返回cur,也就是最后一个节点
		return cur;
	}
}

总结

  1. 链表中,要删除这个某个元素,必须要知道这个元素的前驱元素。
  2. 设计链表的时候需要打牢基础,最好多手写几次,这样对于链表的理解就更加深刻了
  3. 反转链表不难,但是递归还是没有搞懂,需要加强递归的理解。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值