小白都能看懂的力扣算法详解——链表(一)

!!本篇所选题目及解题思路均来自代码随想录 (programmercarl.com)

一 203.移除链表元素

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

203. 移除链表元素 - 力扣(LeetCode)

我们的目标是要寻找val等于目标值的节点,那么我们就要遍历这个链表,找到该节点,之后让该节点的上一个节点指向它的下一个节点,即可实现“删除”。但是我们知道,单向链表只能指向它的下一个节点,那么我们该如何找到它的上一个节点呢?这里我能可以定义一个指针cur,让cur指向该元素的前一个节点,那么cur.next = cur.next.next 即实现了该节点的上一个节点指向它的下一个节点。因此我们可以得出cur.next = head;也就是说cur是头结点的前一个节点。解决了“删除”的问题,接下来思考如何实现寻找这个节点。也就是说,满足val等于目标值这个条件,翻译过来就是cur.next.val == targrt;(别忘了cur是当前节点的前一个节点,cur.next才是该节点)。也就是满足条件时执行cur.next = cur.next.next 操作,不满足条件时cur就移向下一个节点,继续遍历。最后一个问题,如何判断遍历何时结束?不难判断出,当cur指向链表的最后一个节点时,循环就该结束了(此时这个节点的val已经在cur指向它前一个节点时就对比过了),而cur.next为null也无法得到cur.next.val了。也就是说,我们的循环判断条件就是cur.next != null

整体思路结束,接下来来考虑一些特殊点,比如各种临界值。首先是当target==head.val的情况,满足cur.next.val == targrt条件;之后是target等于最后一个节点的值的情况,我们已经谈论过了;第三是链表长度为0的情况,此时head为null,即满足cur.next为null,退出循环;最后是有连续节点满足val等于target的情况,这时我们就会发现,如果有连续值满足条件,那么第二个节点就会被跳过对比,如何解决这个问题呢?我们可以在进行完cur.next = cur.next.next操作后,先不急着将cur向后移动(cur = cur.next),而是让它在原位置不动,这样下一轮比较cur.next.val时,cur.next就是第二个节点啦。

最后瞅一眼返回值,是返回头节点吗?如果直接返回head,那么如果head.val刚好等于target,已经被我们删除了,该怎么办?这时候就要请出我们的老朋友——虚拟头节点了(不熟也没关系,多刷几道链表题目之后就会发现,基本上有链表的地方就有它)。在最开始,设置一个虚拟头节点指向head,即p.next = head;无论头结点被删掉多少个,p.next都会指向它的下一项元素,也就是p始终都是“首个节点”,最后直接返回p.next即可。

class Solution {public ListNode removeElements(ListNode head, int val) {ListNode p = new ListNode();p.next = head;ListNode cur = p;while(cur.next != null) {if(cur.next.val == val) {cur.next = cur.next.next;} else {cur = cur.next;}}return p.next;}}

 代码随想录上提供了另一种不适用虚拟头节点的思路,这里直接附上代码,不再具体讲解。其实思路与上面几乎一致,只是多了一个对头节点是否需要删除的判断,就像上面说过的,如果head.val刚好等于target,已经被我们删除了,该如何找到链表的头部?

class Solution {public ListNode removeElements(ListNode head, int val) {while(head != null && head.val ==val) {head = head.next;}ListNode cru = head;while(cru != null && cru.next != null){if(cru.next.val == val) {cru.next = cru.next.next;} else{cru = cru.next;}}return head;}}

 关于此方法的具体讲解,以及该题目的视频讲解,其他语言版本代码,大家可以到代码随想录中自行查看。代码随想录 (programmercarl.com)icon-default.png?t=N7T8https://www.programmercarl.com/0203.%E7%A7%BB%E9%99%A4%E9%93%BE%E8%A1%A8%E5%85%83%E7%B4%A0.html#%E7%AE%97%E6%B3%95%E5%85%AC%E5%BC%80%E8%AF%BE

二 206.反转链表

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

206. 反转链表 - 力扣(LeetCode)

 这道题乍一看很难(反正我是毫无思路),看到有一种暴力解法是将链表中的元素存储在数组中,反转数组,再放回到链表里,感觉可行但是我没具体尝试(也太暴力了一点......).听完卡尔哥的讲解之后我的感受只有两个字:秒啊。

回归正题,翻转链表我们很自然就能想到让下一个元素指向上一个元素,问题在于,在单向链表中,如何获得上一个元素的位置呢?这里又要用到虚拟头节点了。我们定义一个指针pre指向虚拟头节点,也就是head的前一个元素,再定义一个指针cur指向head,让cur.next指向pre,即可完成下一个元素指向上一个元素,之后我们只要向后移动pre和cur,循环上面的操作即可。是不是很完美?错!这种做法存在一个很明显的错误,当cur.next指向pre之后,head后面的节点就失去了指向它的头节点,如图,那么cur如何如我们所愿向后移动呢?

这时候,我们就需要清楚我们的第三个指针temp出场了。我们在执行cur.next = pre之前,先令p指向cur.next,如此一来,就可保留住cur下一个节点的位置,我们就能随意改变cur.next的指向了。当我们完成cur.next = pre的操作后,只要让cur = temp,就可以让cur指向它原先的下一个节点了。

大体思路完成,接下来思考实现细节。

1.什么时候终止循环?

当循环进行到这一步时,整个反转操作就完成了,再往下走(cur = null)也就没有了意义,所以可以得到,临界条件为cur!=null

2.返回值是什么?

搞不明白就画图,在最后一步操作执行完时,指针情况是这样的,此时cur!=null条件不满足,结束循环。原先的最后一个节点就是现在的头节点,也就是pre指针所指向的节点,因此我们返回pre。

3.链表为空或只有一个节点时,是否需要特殊讨论?

链表为空时,head也为空,此时自然满足cur=null的条件,不进入循环,返回pre,为空。符合。

链表只有头节点时,cur != null,完成第一轮交换,之后pre指向了head,cur执行null,退出循环,返回pre(head),满足。

class Solution {public ListNode reverseList(ListNode head) {ListNode cur = new ListNode();ListNode pre = new ListNode();cur = head;pre= null; while(cur != null ) {ListNode temp = new ListNode();temp = cur.next;cur.next = pre;pre = cur;cur = temp;}return pre;}
}

关于该题目的视频讲解,其他语言版本代码,大家可以到代码随想录中自行查看。

代码随想录 (programmercarl.com)icon-default.png?t=N7T8https://www.programmercarl.com/0206.%E7%BF%BB%E8%BD%AC%E9%93%BE%E8%A1%A8.html

三 707.设计链表

题目描述:

你可以选择使用单链表或者双链表,设计并实现自己的链表。

单链表中的节点应该具备两个属性:val 和 next 。val 是当前节点的值,next 是指向下一个节点的指针/引用。

如果是双向链表,则还需要属性 prev 以指示链表中的上一个节点。假设链表中的所有节点下标从 0 开始。

实现 MyLinkedList 类:

  • MyLinkedList() 初始化 MyLinkedList 对象。
  • int get(int index) 获取链表中下标为 index 的节点的值。如果下标无效,则返回 -1 。
  • void addAtHead(int val) 将一个值为 val 的节点插入到链表中第一个元素之前。在插入完成后,新节点会成为链表的第一个节点。
  • void addAtTail(int val) 将一个值为 val 的节点追加到链表中作为链表的最后一个元素。
  • void addAtIndex(int index, int val) 将一个值为 val 的节点插入到链表中下标为 index 的节点之前。如果 index 等于链表的长度,那么该节点会被追加到链表的末尾。如果 index 比长度更大,该节点将 不会插入 到链表中。
  • void deleteAtIndex(int index) 如果下标有效,则删除链表中下标为 index 的节点。

707. 设计链表 - 力扣(LeetCode)

 这道题非常非常非常麻烦,但是其实难度不是很高,建议大家可以做完其他链表题目后再翻回来做这道题,可能更容易一些。这里直接给出代码及相应注释,不再具体讲解,关于该题目的视频讲解以及其他语言版本代码,大家可以到代码随想录中自行查看。代码随想录 (programmercarl.com)icon-default.png?t=N7T8https://www.programmercarl.com/0707.%E8%AE%BE%E8%AE%A1%E9%93%BE%E8%A1%A8.html#%E7%AE%97%E6%B3%95%E5%85%AC%E5%BC%80%E8%AF%BE

 // 设计链表结构class ListNode {int val;ListNode next;ListNode(){}ListNode(int val) {this.val=val;}
}
// 初始化
class MyLinkedList {//size存储链表元素的个数int size;//虚拟头结点ListNode head;//初始化链表public MyLinkedList() {size = 0;head = new ListNode(0);}//获取第index个节点的数值,注意index是从0开始的,第0个节点就是头结点public int get(int index) {//如果index非法,返回-1if (index < 0 || index >= size) {return -1;}ListNode currentNode = head;//包含一个虚拟头节点,所以查找第 index+1 个节点for (int i = 0; i <= index; i++) {currentNode = currentNode.next;}return currentNode.val;}//在链表最前面插入一个节点,等价于在第0个元素前添加public void addAtHead(int val) {addAtIndex(0, val);}//在链表的最后插入一个节点,等价于在(末尾+1)个元素前添加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 pred = head;for (int i = 0; i < index; i++) {pred = pred.next;}ListNode toAdd = new ListNode(val);toAdd.next = pred.next;pred.next = toAdd;}//删除第index个节点public void deleteAtIndex(int index) {if (index < 0 || index >= size) {return;}size--;if (index == 0) {head = head.next;return;}ListNode pred = head;for (int i = 0; i < index ; i++) {pred = pred.next;}pred.next = pred.next.next;}
}

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://xiahunao.cn/news/2774854.html

如若内容造成侵权/违法违规/事实不符,请联系瞎胡闹网进行投诉反馈,一经查实,立即删除!

相关文章

Android 识别车牌信息

打开我们心爱的Android Studio 导入需要的资源 gradle //开源车牌识别安卓SDK库implementation("com.github.HyperInspire:hyperlpr3-android-sdk:1.0.3")button.setOnClickListener(v -> {Log.d("Test", "");try (InputStream file getAs…

第二届 N1CTF Junior Crypto-junior RSA WP

题目&#xff1a; from Crypto.Util.number import * from secret import flagm bytes_to_long(flag)def gen(bits):while True:a getPrime(bits)b getPrime(bits)c getPrime(bits)p (a << (2*bits)) (b << bits) cq (c << (2*bits)) (a << …

免费文字转语音工具,一款优秀且永久免费的文字转语音工具,同时拥有多种类型男声女声,支持多国语言转换,支持语速调节和下载!

一、软件简介 该工具只有一个功能&#xff0c;就是将输入框内的纯文本内容转换为指定语言的音频&#xff0c;并且可以自由调节语速及音色&#xff08;男声/女声&#xff09;&#xff0c;其内置了多种语音包&#xff0c;包含男声、女声、普通话、粤语以及方言&#xff0c;并且支…

ctfshow-命令执行(web73-web77)

web73 用不了上一题的通用poc了 因为禁用了strlen 但是可以改一个函数自定义一个函数只要是能实现strlen效果即可 cvar_export(scandir(/));exit(0); 根目录下有一个flagc.txt文件 cinclude(/flagc.txt);exit(0); web74 禁用了scandir函数 那就使用web72的glob协议 查看目录下…

如何在vue中使用拖动排序组件sortablejs

效果图&#xff1a; 1.首先&#xff0c;我们需要在vue项目中安装依赖&#xff1a; npm install -save sortablejs2.创建demo文件>demoTest.vue&#xff0c;直接附上实例代码&#xff1a; <template><div><div id"table-names"><div class&…

c#: 表达式树的简化

环境&#xff1a; .net 6 一、问题&#xff1f; 有下面的表达式&#xff1a; var nums new List<int> { 1, 2, 3 }; Expression<Func<int, bool>> exp i > i > nums.Max();我们知道&#xff0c;它其实就是&#xff1a;exp i > i > 3; 那么…

『运维备忘录』之 Ansible 自动化运维工具

一、简介 Ansible是基于Python开发&#xff0c;集合了众多运维工具&#xff08;puppet、cfengine、chef、func、fabric&#xff09;的优点&#xff0c;实现了批量系统配置、批量程序部署、批量运行命令等功能的自动化运维工具&#xff0c;广泛用于配置管理、应用部署以及任务协…

ArcGIS学习(六)地理数据库

ArcGIS学习(六)地理数据库 上个任务我们讲了一个非常重要的知识点一一坐标系。这个任务我们带来另外一个很重要的知识点一一地理数据库。 地理数据库的内容相比于坐标系简单很多! 首先,先让我们来学习下地理数据库的理论。 ArcGIS 中的地理数据库(Geodatabase)是一个用…

牛客网SQL264:查询每个日期新用户的次日留存率

官网链接&#xff1a; 牛客每个人最近的登录日期(五)_牛客题霸_牛客网牛客每天有很多人登录&#xff0c;请你统计一下牛客每个日期新用户的次日留存率。 有一个登录(login。题目来自【牛客题霸】https://www.nowcoder.com/practice/ea0c56cd700344b590182aad03cc61b8?tpId82 …

25、数据结构/二叉树相关练习20240207

一、二叉树相关练习 请编程实现二叉树的操作 1.二叉树的创建 2.二叉树的先序遍历 3.二叉树的中序遍历 4.二叉树的后序遍历 5.二叉树各个节点度的个数 6.二叉树的深度 代码&#xff1a; #include<stdlib.h> #include<string.h> #include<stdio.h> ty…

如何修复Mac的“ kernel_task” CPU使用率过高的Bug?

当计算机开始缓慢运行时&#xff0c;这从来都不是一件有趣的事情&#xff0c;但是当您弄不清它为何如此缓慢时&#xff0c;甚至会变得更糟。如果您已经关闭了所有程序&#xff0c;并且Mac上的所有内容仍然感觉像是在糖蜜中移动&#xff0c;这可能是令人讨厌的kernel_task导致高…

mysql 中文编码问题

前言 最近在学springboot整合mybatisplus技术&#xff0c;用到mysql数据库&#xff0c;然后发现在windows下插入数据表会出现中文乱码现象 (例如 “我是谁” 在数据库中就成了 “???”) windows show variables like %char%;建表时, 设置默认charset为gbk create table u…

3.1-媒资管理之需求分析+搭建Nacos

文章目录 媒资管理模块1 模块需求分析1.1 模块介绍1.2 业务流程1.2.1 上传图片1.2.2 上传视频1.2.3 处理视频1.2.4 审核媒资 2.2 搭建Nacos2.2.1 服务发现中心2.2.2 配置中心2.2.2.1 配置三要素2.2.2.3配置content-api 2.2.3 公用配置2.2.4 配置优先级2.2.5 导入配置文件2.2.6 …

【芯片设计- RTL 数字逻辑设计入门 11 -- 移位运算与乘法】

请阅读【嵌入式开发学习必备专栏 】 文章目录 移位运算与乘法Verilog Codeverilog 拼接运算符&#xff08;{}&#xff09;Testbench CodeVCS 波形仿真 问题小结 移位运算与乘法 已知d为一个8位数&#xff0c;请在每个时钟周期分别输出该数乘1/3/7/8,并输出一个信号通知此时刻输…

15章-Python编程:从入门到实践

第15章生成数据 数据可视化指的是通过可视化表示来探索数据&#xff0c;它与数据挖掘数紧密相关&#xff0c;而数据挖掘指的是使用代码来探索数据集的规律和关联。 数据集可以是用一行代码就能表示的小型数字列表&#xff0c;也可以是数以吉字节的数据。漂亮地呈现数据关乎的并…

Android Studio安装过程遇到SDK无法安装问题解决

首次打开studio遇到该类问题&#xff0c;需要下载SDK文件&#xff0c;后又发现SDK由于是Google源&#xff0c;无法进行正常安装&#xff0c;故转而进行SDK的镜像安装。 一、下载SDK Tools 地址&#xff1a;AndroidDevTools - Android开发工具 Android SDK下载 Android Studio…

Docker实战01

七十八、compse是什么能干嘛 docker-compose容器编排&#xff08;你的容器实例太多了&#xff0c;你如何管理&#xff0c;容器之间涉及到启动的顺序&#xff0c;容器之间涉及到网络通信的调用&#xff09; 1、是什么&#xff1f; Docker-Compose是Docker官方的开源项目&…

【Boost】:http_server模块(六)

http_server模块 一.安装cpp-httplib库二.基本使用服务器 一.安装cpp-httplib库 可以自己写一个http服务器&#xff0c;但比较麻烦&#xff0c;这里直接使用库。 在gitee上搜索cpp-httplib&#xff0c;任意找一个即可&#xff08;建议使用0.7.15版本&#xff09;。例如&#xf…

Qt QVariant类应用

QVariant类 QVariant类本质为C联合(Union)数据类型&#xff0c;它可以保存很多Qt类型的值&#xff0c;包括 QBrush&#xff0c;QColor&#xff0c;QString等等&#xff0c;也能存放Qt的容器类型的值。 QVariant::StringList 是 Qt 定义的一个 QVariant::type 枚举类型的变量&…

Java Stram 流对于返回对象的处理 (结束流)

Java Stram 流对于返回对象的处理 &#xff08;结束流&#xff09; package com.zhong.streamdemo.showdownstreamdemo;import lombok.AllArgsConstructor; import lombok.Data; import lombok.NoArgsConstructor;import java.util.*; import java.util.stream.Collectors; im…