优雅的删除链表元

封面:优雅的删除链表元素.png

王有志,一个分享硬核Java技术的互金摸鱼侠
加入Java人的提桶跑路群:共同富裕的Java人

在数据结构:链表中,我们实现了链表的删除方法,但代码看起来并不“优雅”,那么今天我们就来尝试使用多种方法,“优雅”的实现链表的删除方法。

移除链表元素

今天我们从一道练习题开始:203.移除链表元素。为了方便我们把力扣提供的节点声明抄过来:

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;
	}
}

题目中给定了这样的一个头节点head和待删除元素val,要求删除链表中对应的元素。

“朴素”的删除方式

这道题目很简单,甚至是比数据结构:链表中实现的删除方法还要简单,因为我们不需要维护头节点,尾节点以及链表的规模。
照葫芦画瓢可以得到如下代码:

public static ListNode removeElements(ListNode head, int val) {
	while (head != null && head.val == val) {
		ListNode next = head.next;
		head.next = null;
		head = next;
	}

	if (head == null) {
		return null;
	}

	ListNode prev = head;

	while (prev.next != null) {
		ListNode current = prev.next;
		if (current.val == val) {
			prev.next = current.next;
			current = null;
		} else {
			prev = prev.next;
		}
	}
	return head;
}

空间换“美感”

虽然“朴素”的方法在功能上是没有问题的,但是过多的if语句很让头疼,我们是不是可以减少if语句,使用统一的逻辑去处理呢?
很容易想到的是,我们可以创建新的链表,去承载不需要删除的元素。接下来,我们使用移除链表元素题目中的示例一来描述:

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

图1:空间换“美感”.png
代码如下:

public static ListNode removeElements(ListNode head, int val) {
	if (head == null) {
		return null;
	}
	ListNode newHead = null;
	ListNode current = null;
	while (head != null) {
		if (head.val != val) {
			if (newHead == null) {
				newHead = new ListNode(head.val);
				current = newHead;
			} else {
				current.next = new ListNode(head.val);
				current = current.next;
			}
		}
		head = head.next;
	}
	return newHead;
}

虽然代码只少了1行,但是逻辑却简单了起来,仅仅只需要一个while循环,也只需要处理一次头节点的逻辑即可。
你有没有发现,上面的两种方法,都要对头节点进行特殊处理,是不是“消除”了头节点逻辑会更加简单?
那么我们可以添加一个“有名无实”的头节点,让真实的头节点退下来,这样就可以使用统一的逻辑的处理,代码如下:

public static ListNode removeElements(ListNode head, int val) {
	if (head == null) {
		return null;
	}

	ListNode virtualHead = new ListNode();
	ListNode prev = virtualHead;

	while (head != null) {
		if (head.val != val) {
			prev.next = new ListNode(head.val);
			prev = prev.next;
		}
		head = head.next;
	}
	return virtualHead.next;
}

这样仅代码少了很多,而且代码的逻辑也变得十分清晰了,我们习惯将这种“假的”头节点称为虚拟头节点

虚拟头节点

虚拟头节点是解决链表问题中非常好用的方法,链表中的多种操作都要对头节点进行特殊逻辑处理,这样既不美观,也容易遗漏边界情况,如果没有头节点,处理起来是不是就非常容易了?
图2:虚拟头节点.png
在王争老师的《数据结构与算法之美》中,也提到了这种方法,他称虚拟头节点为“哨兵”,并将含有“哨兵”的链表称为“带头链表”,当然无论叫什么,其实现思想都是相同的。
上面的实现的删除操作中,我们讨了一个巧,在没有要求空间复杂度的情况下,我们从删除变成了“新增”,空间复杂度是 O ( n ) O(n) O(n),如果我们要求空间复杂度为 O ( 1 ) O(1) O(1)呢?

private static ListNode removeElements(ListNode head, int val) {
	ListNode virtualHead = new ListNode();
	virtualHead.next = head;
	ListNode prev = virtualHead;

	while (prev.next != null) {
		ListNode current = prev.next;
		if (current.val == val) {
			prev.next = current.next;
			current.next = null;
		} else {
			prev = prev.next;
		}
	}
	return virtualHead.next;
}

递归:优雅永不过时

我们通过虚拟头节点,成功实现了移除头节点的特殊逻辑,减少了代码量,使得结构更加清晰,不过有没有更加“优雅”的解法?
显然是有的,还记得我们在[[04.编程技巧:从斐波那契数列到递归]]中引用邓俊峰老师的话吗?其中有一句:

从程序结构的角度看,递归模式能够统筹纷繁多变的具体情况,避免复杂的分支以及嵌套的循环,从而更为简明地描述和实现算法,减少代码量,提高算法的可读性,保证算法的整体效率。

另外我们也提到了,递归是将复杂庞大的问题不断的拆分成更小的问题,逐一解决后合并结果。拆分的动作是递,合并的动作是归,结合起来便是递归。
在这道题目中我们如何去拆分问题?可以将链表拆分成子链表,再进行逻辑处理:
图3:递归拆分链表.png
这就是我们拆分链表的过程,也是递归中最重要的部分,将大规模问题拆分成基础问题
在这道题目中,最基础的问题是什么?是链表中的节点是否符合删除的要求,那么求解就非常简单了:

private static ListNode removeElements(ListNode node, int val) {
	if(node.val == val) {
		return node.next;
	}else {
		return node;
	}
}

好了,我们有了拆分的过程,也对基础问题进行了求解,最后只需要合并问题的结果了,合并的过程也很简单。
图4:递归合并链表.png
每次都会从输入链表中拆分出子链表,合并时,只需要将处理过后的子链表作为上一次调用的后继节点即可,代码如下:

prevNode.next = removeElements(prevNode.next, val);

这样我们再处理下边界情况,将两部分结合起来,就可以写出完整的代码了:

private static ListNode removeElements(ListNode head, int val) {
	if(head == null) {
		return null;
	}

	head.next = removeElements(head.next, val);
	return head.val == val ? head.next : head;
}

我们再来分析下递归的复杂度,时间复杂度 O ( n ) O(n) O(n)是一定的了,链表的特性就决定了牵扯到查找,时间复杂度就一定为 O ( n ) O(n) O(n),那么空间复杂度呢?
从纸面上看,没有引入任何临时变量,不需要额外的空间,堪称完美。但是不要忘记递归是方法调用,那么就需要调用栈,隐性的开销还是会比较大。

再谈递归

说真的,第一次见到递归解法时,我是一头雾水。直到动手画出的过程,才理解递归的解法多么巧妙,不禁感叹前人的智慧。
这也让我想起来Aditya Bhargava在《算法图解》中说到的:

递归是我最喜欢的主题之一,它将人分成三个截然不同的阵营:恨它的、爱它的以及恨了几年后又爱上它的。

现在,我想我是爱上递归了。
言归正传,我们来总结下实现递归都需要哪些要素:

  • 问题可以拆分为规模更小的基础问题;
  • 基础问题的解法与原始问题一致;
  • 递归存在出口条件。

而递归的过程也可以拆分为3个阶段:

  • 不断拆分问题,就是递的过程
  • 解决基础问题
  • 合并问题的解,这是归的过程

虽然递归实现了“终极优雅”,但在使用过程中还是要注意两个问题:重复计算调用栈的消耗

结语

今天我们对单向链表的删除方法不断的优化,减少复杂逻辑,减少代码量。
介绍了使用虚拟头节点处理链表中的问题,成功的消除了特殊逻辑,然后是通过递归,实现了“终极优雅”。
最后,我们第二次聊到了递归,当然这不会是我们最后一次聊递归,在后面树的内容中,递归会更加频繁的出现。

练习

  • 203.移除链表元素
  • 234.回文链表
  • 876.链表的中间结点
  • 剑指 Offer 25.合并两个排序的链表
  • 剑指 Offer 62.圆圈中最后剩下的数字

如果本文对你有帮助的话,还请多多点赞支持。如果文章中出现任何错误,还请批评指正。最后欢迎大家关注分享硬核Java技术的金融摸鱼侠王有志,我们下次再见!

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【深度学习:Micro-Models】用于标记图像和视频的微模型简介

【深度学习:Micro-Models】用于标记图像和视频的微模型简介 微模型:起源故事微模型到底是什么?更详细地解释微观模型:一维标签蝙蝠侠效率 在计算机视觉项目中使用微模型的额外好处面向数据的编程 在本文中,我们将介绍 …

SystemC学习笔记 - Hello systemc world

Hello Systemc World 码农老规矩,先写一个hello world并输出,语法什么的后面再说,先能编译运行再说。 目录配置 使用examples里的配置,在examples/sysc目录下创建test目录,其下创建第一个test1的目录,如…

【Py/Java/C++三种语言详解】LeetCode每日一题240114【链表】LeetCode83、删除排序链表中的重复节点

文章目录 题目链接题目描述解题思路代码PythonJavaC时空复杂度 华为OD算法/大厂面试高频题算法练习冲刺训练 题目链接 LeetCode83、删除排序链表中的重复节点 题目描述 给定一个已排序的链表的头 head , 删除所有重复的元素,使每个元素只出现一次 。返…

寡年是否适合结婚?寡妇年结婚有什么禁忌吗?让程序来告诉你有多少人是寡妇年结婚的。

什么是寡年? 百度百科 原文:寡年-百度百科 指整年没有“立春”的日子就是“盲年”,俗称寡年。又名滑头年 社会上流传的“寡妇年”,是指整个农历年都没有立春的年份。以农历2005年的鸡年为例,立春在公历2月4日&…

8年老鸟,自动化测试经验,测试数据管理分析总结,一篇打通...

目录:导读 前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结(尾部小惊喜) 前言 问题:…

Linux 转换文字编码与换行符 nkf命令

参考资料 【 nkf 】コマンド――文字コードと改行コードを変換するnkfコマンドでファイルの文字コードと改行コードを統一する 目录 一. 前期准备二. 乱码现象与分析三. nkf命令3.1 nkf --guess 查看文件编码3.2 nkf -w8 UTF-8(BOM)编码显示3.3 nkf -wd --overwrite 覆盖源文件…

关于CodeReview的一些实践和思考

在日常开发中,Code Review 的重要性日益凸显。它不仅有助于提升代码质量,还促进了团队成员之间的知识共享和技能提升。本文将主要聚焦于 Code Review,分享在这个过程中的一些心得和思考。 CodeReview常用到的一些术语 之前看到公司的大佬经…

项目进度管理

7过程 计划过程组6项:规划进度管理,定义活动,排列活动顺序,估算活动资源,估算活动持续时间,制定进度计划, 监控过程组1项:控制进度 1、规划进度管理, 对项目过程中管理…

Webpack模块打包工具

目录 Webpack模块打包工具知识点自测01.Webpack 简介以及体验目标讲解小结 02.Webpack 修改入口和出口目标讲解小结 03.案例-用户登录-长度判断目标讲解小结 04.Webpack 自动生成 html 文件目标讲解小结 05.Webpack-打包 css 代码目标讲解小结 06.优化-提取 css 代码目标讲解小…

C++学习笔记——继承和动态内存分配

目录 一、继承 二、动态内存分配 三、继承的细节 四、动态内存分配细节 五、一个动物园管理系统 继承和动态内存分配是C中两个重要的概念 一、继承 继承是C中面向对象编程的一个重要特性,它允许我们创建一个新类,该类从现有的类中继承属性和方法&…

设置ubuntu命令行样式

目录 一、脚本 二、含义 三、颜色设置 四、展示 五、注意 上次为了学习ros安装了一个22.04并且做了简单的配置,这次我们进一步对命令行样式进行配置 ubuntu22.04安装与配置_ubuntu22.04硬件配置-CSDN博客 一、脚本 这是他的默认配置,太长了&#x…

Python--循环语句

在 Python 中,循环语句用于重复执行一段代码多次。Python 主要提供了两种类型的循环:for 循环和 while 循环。 1. for 循环 for 循环用于遍历可迭代对象(如列表、元组、字典、字符串等)中的每个元素,并对每个元素执行…

【每日一题】构造限制重复的字符串

文章目录 Tag题目来源解题思路方法一:贪心空间复杂度: O ( ∑ ) O(\sum) O(∑)。 写在最后 Tag 【贪心】【字符串】【2024-01-13】 题目来源 2182. 构造限制重复的字符串 解题思路 方法一:贪心 思路 解题思想比较简单,利用贪…

C++进阶--红黑树

红黑树 一、红黑树的概念二、红黑树的性质三、红黑树结点的定义四、红黑树的插入五、红黑树的验证七、红黑树的查找七、红黑树与AVL树的比较七、完整代码RBTree.h 一、红黑树的概念 红黑树,是一种二叉搜索树,但在每个结点上增加一个存储位表示结点的颜色…

IaC基础设施即代码:使用Terraform 连接 alicloud阿里云

目录 一、实验 1.环境 2.alicloud阿里云创建用户 3.Linux使用Terraform 连接 alicloud 4.Windows使用Terraform 连接 alicloud 二、问题 1.Windows如何申明RAM 相关变量 2.Linux如何申明RAM 相关变量 3. Linux terraform 初始化失败 4.Linux terraform 计划与预览失败…

关于高通Android 平台上qssi的介绍

1. QSSI 是 Qualcomm Single System Image 的缩写。 2. Android Q上开始支持QSSI。 3. QSSI 是用来编译system.img的 3.1 QSSI编译注意事项 lunch qssi ------ 编译system.img lunch target ------ 编译其余的image 3.2 有QSSI和没有QSSI的编译流程对比 没有QS…

YOLOv5独家原创改进:多层次特征融合(SDI)结合PConv、DualConv、GSConv,实现二次创新 | UNet v2最新论文

💡💡💡本文独家改进:多层次特征融合(SDI)高效结合DualConv、PConv、GSConv等实现二次创新 1)替代原始的Concat; 收录 YOLOv5原创自研 https://blog.csdn.net/m0_63774211/category_12511931.html 💡💡💡全网独家首发创新(原创),适合paper !!! 💡�…

Docker运行RabbitMQ并使用SpringAMQP操作

文章目录 一、RabbitMQ运行二、整合SpringAMQP1. 引入依赖 三、测试1. 消费者2. 生产者3. 运行 一、RabbitMQ运行 拉取docker镜像 docker pull rabbitmq:3-management基础运行命令 docker run \-e RABBITMQ_DEFAULT_USERrabbitmq \-e RABBITMQ_DEFAULT_PASSrabbitmq \--name…

Vue3 如何使用移动端调试工具vConsole

1、安装 pnpm i vconsole2、在src/utils下新建vconsole.ts,写入以下代码 // 这是移动端控制台调试工具,需要调试就打开,不用就注释 import vConsole from vconsole const vconsole new vConsole()3、src/main.ts 引入,需要调试就打开,&…

数据结构学习之链式栈应用的案例(最小栈)

实例要求: 设计一个支持入栈、出栈、取栈顶元素等操作,并能在常数时间内检索到最小元素的栈; 实现 MinStack 类: MinStack* minStackCreate() 初始化堆栈对象,即建栈; void minStackPush(MinStack* obj, int val) …