20230421 | 203. 移除链表元素、707. 设计链表、206. 反转链表

1、203. 移除链表元素

在这里插入图片描述

方法1:不添加虚拟节点方式,但是要注意处理删除头部的数据
时间复杂度 O(n)
空间复杂度 O(1)

/**
 * 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) {
        //处理全部一样的
        while(head!=null && head.val == val){
            head = head.next;
        }
        //已经为null,提前退出
        if(head == null) return head; 
        //前驱节点
        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;
    }
}

方法二:增加一个虚拟头节点,可以方便删除头部的数据,但主要返回时的位置是什么,构建头节点的时候要注意是:ListNode dump = new ListNode(-1,head); 或者是

在这里插入图片描述

public ListNode removeElements(ListNode head, int val) {
        //已经为null,提前退出
        if(head == null) return head; 
        ListNode dump = new ListNode(-1,head);
        //前驱节点
        ListNode pre = dump;
        //当前节点
        ListNode cur = head;
        while(cur!=null){
            if(cur.val == val){
                pre.next = cur.next; //前驱节点的指针指向当前节点的尾部
            }else{
                pre = cur; //前驱节点指向当前节点
            }
            cur = cur.next; //当前节点往后移动
        }
        return dump.next;
    }

2、707. 设计链表

在这里插入图片描述
删除链表节点: 链表-删除节点
在这里插入图片描述

添加链表节点: 链表-添加节点
在这里插入图片描述

这道题目设计链表的五个接口:

获取链表第index个节点的数值
在链表的最前面插入一个节点
在链表的最后面插入一个节点
在链表第index个节点前面插入一个节点
删除链表的第index个节点

可以说这五个接口,已经覆盖了链表的常见操作,是练习链表操作非常好的一道题目

链表操作的两种方式:

直接使用原来的链表来进行操作。
设置一个虚拟头结点在进行操作。
class ListNode{
    int val; //值
    ListNode next;
    ListNode(){}
    ListNode(int val){
        this.val = val;
    }
}
class MyLinkedList {
    //size 存储链表元素的个数
    int size;
    //虚拟节点
    ListNode head;

    //初始化 MyLinkedList 对象。
    public MyLinkedList() {
        size =0;
        head = new ListNode(0);
    }
    
    //获取第index个节点的数值,注意index是从0开始的,第0个节点就是头节点
    public int get(int index) {
        //如果下标无效,则返回 -1 。
        if(index<0|| index>=size){
            return -1;
        }
        ListNode currNode = head; //当前
        //包含一个虚拟头节点,所以查找第index+1个节点
        for(int i=0;i<=index;i++){
            currNode = currNode.next;
        }
        //返回值
        return currNode.val;
    }
    
    // 将一个值为 val 的节点插入到链表中第一个元素之前。在插入完成后,新节点会成为链表的第一个节点。
    // 等价于在第0个元素前添加
    public void addAtHead(int val) {
        addAtIndex(0,val); //借助按索引位置添加
    }
    
    //将一个值为 val 的节点追加到链表中作为链表的最后一个元素。
    //等价于在size+1的位置添加
    public void addAtTail(int val) {
        addAtIndex(size,val);
    }
    
    //将一个值为 val 的节点插入到链表中下标为 index 的节点之前。如果 index 等于链表的长度,那么该节点会被追加到链表的末尾。如果 index 比长度更大,该节点将 不会插入 到链表中。
    public void addAtIndex(int index, int val) {
        if(index>size) return;
        if(index<0) index = 0;
        size++;//链表长度+1
        
        //找到要插入节点的前驱
        ListNode pre = head;
        for(int i=0;i<index;i++){
            pre = pre.next;
        }
        ListNode toAdd = new ListNode(val); //创建节点
        toAdd.next = pre.next; //创建的节点指向插入的位置
        pre.next = toAdd; //指向当前节点
    }
    //如果下标有效,则删除链表中下标为 index 的节点。
    public void deleteAtIndex(int index) {
        if(index<0 || index>=size){
            return;
        }
        size--;
        if(index == 0) { //删除头节点的情况
            head = head.next;
            return;
        }
        //先找打前驱
        ListNode pre = head;
        for(int i=0;i<index;i++){
            pre = pre.next;
        }
        pre.next = pre.next.next; //指向改变,实现删除
    }
}

/**
 * Your MyLinkedList object will be instantiated and called as such:
 * MyLinkedList obj = new MyLinkedList();
 * int param_1 = obj.get(index);
 * obj.addAtHead(val);
 * obj.addAtTail(val);
 * obj.addAtIndex(index,val);
 * obj.deleteAtIndex(index);
 */

需要反复练习

3、206. 反转链表

请添加图片描述
在这里插入图片描述

/**
 * 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) {
        if(head ==null || head.next ==null) return head;
        ListNode cur = head;//当前节点
        ListNode pre = null;//前驱节点
        while(cur!=null){
            ListNode temp = cur.next; //记录下一个节点的位置,防止找不到
            cur.next = pre; //第一次pre为空,实现链表的端口操作
            pre = cur; //前驱节点指向下一个
            cur = temp; //当前节点指向临时存储的节点(下一个节点)
        }
        return pre; //返回pre节点,是因为pre节点已经指向最后一个位置了,各自的指针已经转变,并且此时cur为null。我错误的以为是head,其实head只指向1一个值了
    }
}

注意: 返回pre节点,是因为pre节点已经指向最后一个位置了,各自的指针已经转变,并且此时cur为null。我错误的以为是head,其实head只指向1一个值了

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值