首页 > 其他分享 >代码随想录|day3 链表 203.移除链表元素、707.设计链表、206.反转链表

代码随想录|day3 链表 203.移除链表元素、707.设计链表、206.反转链表

时间:2024-11-01 20:46:36浏览次数:3  
标签:head ListNode cur int 随想录 next 链表 移除 size

基础知识:

代码随想录

203.移除链表元素

建议: 本题最关键是要理解 虚拟头结点的使用技巧,这个对链表题目很重要。

这里主要记录用虚头的方法。即设置一个虚拟的头指针帮忙解题。

先看代码:

class Solution {
    public ListNode removeElements(ListNode head, int val) {
	ListNode dummy=new ListNode();
	dummy.next=head;

	ListNode curr=dummy;
	while(curr.next!=null)
	{
		if (curr.next.val==val)
		{
			curr.next=curr.next.next;
		}
		else
		{
			curr=curr.next;
		}
	}
	return dummy.next;
	}
}

这样做有一个很明确的目的:因为我们是通过前一个节点来判断后一个节点是否该删除,但是头节点没有前一个节点,所以我们可以虚拟一个新的头节点,这样真实的头节点也有了头节点,我们的代码就不需要写一个if来判断是否为头节点这种情况。

707.设计链表

这个其实可以当作对刚才所学的一个复习,实现可以分为单指针和双指针

以下为单链表:
class MyLinkedList {

        int size;
        ListNode head;
    public MyLinkedList() {
         size=0;
         head=new ListNode(0);
        
    }
    
    public int get(int index) {
        if (index<0||index>=size)
        {
            return -1;
        }
        ListNode curr=head;
        for (int i=0;i<=index;i++)
        {
            curr=curr.next;
        }
        return curr.val;


    }
    
    public void addAtHead(int val) {
        ListNode newNode=new ListNode(val);
        newNode.next=head.next;
        head.next=newNode;
        size++;

    }
    
    public void addAtTail(int val) {
        ListNode newNode=new ListNode(val);
        ListNode curr=head;
        while (curr.next!=null)
        {
            curr=curr.next;
        }
        curr.next=newNode;
        size++;
    }
    
    public void addAtIndex(int index, int val) {
        if (index>size)
        {
            return;
        }
        if (index<0)
        {index=0;}
        size++;
        ListNode newNode=new ListNode(val);
        ListNode curr=head;
        for (int i=0;i<index;i++)//要在该索引的位置前一个索引才对
        {
         curr=curr.next;
        }
        newNode.next=curr.next;
        curr.next=newNode;

    }
    
    public void deleteAtIndex(int index) {
        if (index < 0 || index >= size) {
            return;
        }
        size--;
        ListNode curr=head;
        for (int i = 0; i < index ; i++) {
            curr = curr.next;
        }
        curr.next=curr.next.next;
    }
   
 }
以下为双链表:
//双链表
class ListNode{
    int val;
    ListNode next,prev;
    ListNode() {};
    ListNode(int val){
        this.val = val;
    }
}


class MyLinkedList {  

    //记录链表中元素的数量
    int size;
    //记录链表的虚拟头结点和尾结点
    ListNode head,tail;
    
    public MyLinkedList() {
        //初始化操作
        this.size = 0;
        this.head = new ListNode(0);
        this.tail = new ListNode(0);
        //这一步非常关键,否则在加入头结点的操作中会出现null.next的错误!!!
        head.next=tail;
        tail.prev=head;
    }
    
    public int get(int index) {
        //判断index是否有效
        if(index>=size){
            return -1;
        }
        ListNode cur = this.head;
        //判断是哪一边遍历时间更短
        if(index >= size / 2){
            //tail开始
            cur = tail;
            for(int i=0; i< size-index; i++){
                cur = cur.prev;
            }
        }else{
            for(int i=0; i<= index; i++){
                cur = cur.next; 
            }
        }
        return cur.val;
    }
    
    public void addAtHead(int val) {
        //等价于在第0个元素前添加
        addAtIndex(0,val);
    }
    
    public void addAtTail(int val) {
        //等价于在最后一个元素(null)前添加
        addAtIndex(size,val);
    }
    
    public void addAtIndex(int index, int val) {
        //index大于链表长度
        if(index>size){
            return;
        }

        size++;
        //找到前驱
        ListNode pre = this.head;
        for(int i=0; i<index; i++){
            pre = pre.next;
        }
        //新建结点
        ListNode newNode = new ListNode(val);
        newNode.next = pre.next;
        pre.next.prev = newNode;
        newNode.prev = pre;
        pre.next = newNode;
        
    }
    
    public void deleteAtIndex(int index) {
        //判断索引是否有效
        if(index>=size){
            return;
        }
        //删除操作
        size--;
        ListNode pre = this.head;
        for(int i=0; i<index; i++){
            pre = pre.next;
        }
        pre.next.next.prev = pre;
        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);
 */

206.反转链表

有两种写法

第一种:双指针

通过循环和两个指针把链表的指向全部反转过来。

加入temp指针的目的是保存cur的next值,这样在你把cur.next赋值给prev时,这个值不会丢失,然后更新变量进入下一个循环

  public ListNode reverseList(ListNode head) {
        ListNode cur=head;
		ListNode prev=null;
		ListNode temp=null;

		while(cur!=null)
		{
			temp=cur.next;
			cur.next=prev;
			prev=cur;
			cur=temp;
		}
		return prev;

    }
}
第二种:递归

其实递归和这个双指针写法是一样一样的,只要理解了双指针解法,该方法非常好理解并且简单。

	public ListNode reverseList(ListNode head) {
		return reverse(null, head);
	}
	public ListNode reverse(ListNode prev,ListNode cur)
	{
		if (cur==null)
		{
			return prev;
		}
		ListNode temp=null;
		temp=cur.next;
		cur.next=prev;
		// 更新prev、cur位置
		// prev = cur;
		// cur = temp;
		return reverse(cur,temp);

	}
}

标签:head,ListNode,cur,int,随想录,next,链表,移除,size
From: https://blog.csdn.net/indexyjk/article/details/143430019

相关文章

  • LeetCode23:合并K个升序链表
    原题地址:.-力扣(LeetCode)题目描述给你一个链表数组,每个链表都已经按升序排列。请你将所有链表合并到一个升序链表中,返回合并后的链表。示例1:输入:lists=[[1,4,5],[1,3,4],[2,6]]输出:[1,1,2,3,4,4,5,6]解释:链表数组如下:[1->4->5,1->3->4,2->6]将它们......
  • 代码随想录算法训练营第三十三天|Day33 动态规划
    62.不同路径https://programmercarl.com/0062.%E4%B8%8D%E5%90%8C%E8%B7%AF%E5%BE%84.html视频讲解:https://www.bilibili.com/video/BV1ve4y1x7Eu思路int**initDP(intm,intn){int**dp=(int**)malloc(sizeof(int*)*m);inti,j;for(i=0;i<......
  • 链表和数组的区别
    链表和数组是两种常用的数据结构,本文旨在详细比较链表和数组的区别包括:1.存储方式不同;2.内存利用不同;3.访问元素的效率不同;4.插入和删除操作的效率不同;5.内存分配的灵活性不同;6.应用场景不同。通过这些比较,读者将更深入地理解两者的特点,以及它们在不同应用场景下的最佳使用方法。......
  • 单链表题+数组题(快慢指针和左右指针)
    @目录说明:本文章用于“单链表题+数组题”“链表”知识一、案例说明(使用快慢指针)问题1.1判断链表是否有环问题1.2:已知链表有环,请返回这个环的起点位置问题1.3:寻找无环单链表的中点,要求:如果偶数个数以左面一个节点为中点问题1.4:寻找无环单链表的中点,要求:如果偶数个数以右面一个节......
  • 代码随想录刷题学习日记
    仅为个人记录复盘学习历程,解题思路来自代码随想录代码随想录刷题笔记总结网址:代码随想录404.左叶子之和计算给定二叉树的所有左叶子之和。(所有的左边的叶子节点的和)提供参数:根结点root关键思路:遍历,判断若为左叶子节点,则将值累加。主要操作:递归三要素1.返回值类型和参......
  • 02链表算法/代码随想录
    前几天忙比赛,算法停了三天,继续开刷,不能停!!二、链表2.1删除链表中的元素两种方案无哨头:要删除节点的前一个结点指向删除节点的指向节点。头节点需要单独定义有哨头:头节点不需要单独定义实战力扣203/***Definitionforsingly-linkedlist.*publicclassLis......
  • 链表(数据结构)
    一.单链表1.1概念与结构再上一篇中我们讲到顺序表,但是顺序表也是有很多的问题,像申请的空间过多过少或者增容该才能不浪费空间,今天我们就来认识一个新的知识,叫做链表,链表也是线性表的一种,链表是一种物理存储结构上非连续、非顺序的存储结构,数据结构的逻辑顺序是通过链表中的......
  • Leetcode21:合并两个有效链表
    原题地址:.-力扣(LeetCode)题目描述将两个升序链表合并为一个新的 升序 链表并返回。新链表是通过拼接给定的两个链表的所有节点组成的。 示例1:输入:l1=[1,2,4],l2=[1,3,4]输出:[1,1,2,3,4,4]示例2:输入:l1=[],l2=[]输出:[]示例3:输入:l1=[],l2=......
  • 代码随想录一刷Day4
    59.螺旋矩阵II思路:找模式:1.从左到右,从上到下,从右到左,从下到上2.转几圈3.注意跟二分一样,统一原则4.注意for里面的循环条件54.螺旋矩阵思路:不能套用螺旋矩阵2 如果在此上进行修改,会漏很多情况动态移动上下边界  注意边界条件,这个需要<=,推一下便知 后面两题前缀......
  • 相交链表
    两个链表的第一个公共结点(相交链表)题目链接:牛客||LeetCode160描述输入两个无环的单向链表,找出它们的第一个公共结点,如果没有公共节点则返回空。(注意因为传入数据是链表,所以错误测试数据的提示是用其他方式显示的,保证传入数据是正确的)例如,输入{1,2,3},{4,5},{6,7}时,两个无......