无头单向非循环链表实现 and leetcode刷题
无头单向非循环链表实现
- 1. 单链表的模拟实现
- IList.java接口:
- MySingleList.java文件:
- 2. leetcode刷题
- 2.1 获取链表的中间节点
- 2.2 删除链表中所有值为value的元素
- 2.3 单链表的逆置
- 2.4 获取链表倒数第k个节点
- 2.5 给定 x, 把一个链表整理成前半部分小于 x, 后半部分大于等于 x 的形式
- 2.6 判定链表是否是回文
- 2.7 判定链表相交并求出交点
- 2.8 判断链表带环
- 2.9 求环的入口点
- 2.10 合并两个有序链表
写在最前面,学习数据结构一定要结合画图!先画图分析,写出伪代码,再仔细分析伪代码是否成立,成立再写入题目中检验!
1. 单链表的模拟实现
单链表的模拟实现需要创建三个文件:IList.java接口文件,MySingleList.java文件,还有一个test.java测试文件。测试文件这里就不演示了。
IList.java接口:
public interface IList {// 1、无头单向非循环链表实现//头插法void addFirst(int data);//尾插法void addLast(int data);//任意位置插入,第一个数据节点为0号下标void addIndex(int index,int data);//查找是否包含关键字key是否在单链表当中boolean contains(int key);//删除第一次出现关键字为key的节点void remove(int key);//删除所有值为key的节点void removeAllKey(int key);//得到单链表的长度int size();void clear();void display();
}
MySingleList.java文件:
public class MySingleList implements IList{static class ListNode {public int val;public ListNode next;public ListNode(int val) {this.val = val;}}public ListNode head;public void creatList() {ListNode node1 = new ListNode(0);ListNode node2 = new ListNode(1);ListNode node3 = new ListNode(2);ListNode node4 = new ListNode(3);node1.next = node2;node2.next = node3;node3.next = node4;head = node1;}@Overridepublic void display() {ListNode cur = head;while (cur != null) {System.out.print(cur.val+" ");cur = cur.next;}System.out.println();}@Overridepublic void addFirst(int data) {ListNode node = new ListNode(data);node.next = head;head = node;}@Overridepublic void addLast(int data) {ListNode node = new ListNode(data);if(head == null) {head = node;return;}//找尾ListNode cur = head;while(cur.next != null) {cur = cur.next;}cur.next = node;}@Overridepublic void addIndex(int index, int data) {if(index < 0 || index > size()) {System.out.println("index位置不合法!");return;}if(index == 0) {addFirst(data);return;}if(index == size()){addLast(data);return;}ListNode node = new ListNode(data);ListNode cur = head;for (int i = 0; i < index-1; i++) {cur = cur.next;}node.next = cur.next;cur.next = node;}@Overridepublic boolean contains(int key) {ListNode cur = head;while(cur != null) {if(cur.val == key) {return true;}cur = cur.next;}return false;}@Overridepublic void remove(int key) {if(head == null) {return;}if(head.val == key) {head = head.next;return;}ListNode pre = head;while(pre.next != null) {if(pre.next.val == key) {ListNode del = pre.next;pre.next =del.next;return;}pre = pre.next;}}@Overridepublic void removeAllKey(int key) {if(head == null) {return;}ListNode pre = head;ListNode cur = head.next;while(cur != null) {if(cur.val == key) {pre.next = cur.next;}else {pre = cur;}cur = cur.next;}//该if语句只能放在最后面,如果头节点需要删除,//删除后有可能下一个节点(此时这个节点做头节点)依然是需要删除的//因此,只能放在最后,当后面的都删除好了,再检查头节点是否需要删除if(head.val == key) {head = head.next;}}@Overridepublic int size() {int len = 0;ListNode cur = head;while(cur != null) {cur = cur.next;len++;}return len;}@Overridepublic void clear() {head = null;}
}
2. leetcode刷题
2.1 获取链表的中间节点
题目链接:876. 链表的中间结点
注意:题目中说明当链表只有一个中间结点时,返回该节点;而当该链表有两个中间结点,返回第二个结点
解析:定义一对“快慢指针”,“快指针”为fast,一次走两步;“慢指针”为slow,一次走一步。
- 当链表的结点个数为奇数个时,fast走到fast.next == null时,slow此时所在位置就是中间节点
- 当链表的节点个数为偶数个时,fast走到fast == null时,slow此时所在位置就是中间节点
代码如下:
class Solution {public ListNode middleNode(ListNode head) {ListNode fast = head;ListNode slow = head;while(fast != null && fast.next != null) {fast = fast.next.next;slow = slow.next;}return slow;}
}
2.2 删除链表中所有值为value的元素
题目链接:203. 移除链表元素
这题的题解和模拟实现单链表的removeAllKey是一样的,故不再赘述。
代码如下:
class Solution {public ListNode removeElements(ListNode head, int val) {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;}if(head.val == val) {head = head.next;}return head;}
}
2.3 单链表的逆置
题目链接:206. 反转链表
解析:只需将链表的每个箭头调转方向即可,即修改当前节点的next值为前一个节点的地址,修改后就无法获取下一个节点了,故需要一个curN来定位下一个节点,又由于是单链表,无法得到前一个节点的位置,所以还需要定义一个prev来定位前一个节点的位置

代码如下:
class Solution {public ListNode reverseList(ListNode head) {if(head == null || head.next == null) {return head;}ListNode pre = head;ListNode cur = head.next;ListNode curN = head.next.next;pre.next = null;while(cur != null) {cur.next = pre;pre = cur;cur = curN;if(curN != null) {curN = curN.next;}}return pre;}
}
2.4 获取链表倒数第k个节点
题目链接:面试题 02.02. 返回倒数第 k 个节点
解析:定义一对“快慢指针”,”快指针“fast先走k步,然后”快指针“fast和”慢指针“slow一起一次走一步,直至fast == null结束,这时slow指向的便是倒数第k个节点
代码如下:
class Solution {public int kthToLast(ListNode head, int k) {if(head == null) {return -1;}ListNode fast = head;ListNode slow = head;for(int i = 0; i < k; i++) {fast = fast.next;}while(fast != null) {fast = fast.next;slow = slow.next;}return slow.val;}
}
2.5 给定 x, 把一个链表整理成前半部分小于 x, 后半部分大于等于 x 的形式
题目链接:CM11 链表分割
注意:这题是将所有小于x的结点排在其余结点之前,且不能改变原来的数据顺序
public class Partition {public ListNode partition(ListNode head, int x) {// write code hereif(head == null) {return null;}ListNode bs = null;//bs:beforestartListNode be = null;//be:beforeendListNode as = null;//as:afterstartListNode ae = null;//ae:afterendListNode cur = head;while(cur != null) {if(cur.val < x) {if(bs == null) {//找到bs和be的起始位置bs = be = cur;}else {be.next = cur;be = cur;}}else {if(as == null) {//找到as和ae的起始位置as = ae = cur;}else {ae.next = cur;ae = cur;}}cur = cur.next;}//ae的next需要手动置为nullif(ae != null) {ae.next = null;}//如果链表的节点都大于x,则返回asif(bs == null) {return as;}//bs不为null,be自然也不为空be.next = as;return bs;}
}
2.6 判定链表是否是回文
题目链接:OR36 链表的回文结构
public class PalindromeList {public boolean chkPalindrome(ListNode A) {// write code hereif(A == null) {return false;}if(A.next == null) {return true;}//1.找到中间节点ListNode mid = getMiddleNode(A);//2.反转后半部分ListNode as = reseverList(mid);mid.next = null;//一定要置null!//3.从前往后依次对比两个链表的val值是否相同ListNode bs = A;while(bs.next != null && as.next != null) {if(bs.val != as.val) {return false;}bs = bs.next;as = as.next;}if(bs.val != as.val) {return false;}return true;}private ListNode reseverList(ListNode head) {ListNode prev = head;ListNode cur = head.next;ListNode curN = cur.next;while(cur != null) {cur.next = prev;prev = cur;cur = curN;if(curN != null){curN = curN.next;}}return prev;}private ListNode getMiddleNode(ListNode A) {ListNode fast = A;ListNode slow = A;while(fast != null && fast.next != null) {fast = fast.next.next;slow = slow.next;}return slow;}
}
2.7 判定链表相交并求出交点
题目链接:160. 相交链表
解题思路:
- 分别求出两个链表的长度,并得到两链表的长度差值(正数)
- 先让长链表的“l指针”走长度差值步,再让“l指针”和“s指针”一起走,如果相遇,相遇点即为相交链表的交点,如果没有相遇,则最后l和s同时为null
- 检验当两个链表同时为null时,代码是否满足
代码如下:
public class Solution {public ListNode getIntersectionNode(ListNode headA, ListNode headB) {int lenA = size(headA);int lenB = size(headB);//先假设headA链表的长度大于headB链表ListNode l = headA;ListNode s = headB;int len = lenA-lenB;//如果是headB链表更长,则进入if语句,进行调换if(len < 0) {len = -len;l = headB;s = headA;}for(int i = 0; i < len; i++) {l = l.next;}while(l != s) {l = l.next;s = s.next;}if(l == null) {return null;//没相交}return l;}public int size(ListNode head) {int len = 0;ListNode cur = head;while(cur != null) {cur = cur.next;len++;}return len;}
}
2.8 判断链表带环
题目链接:141. 环形链表
解题思路:
- 定义一对“快慢指针”,快指针fast一次走两步,慢指针slow一次走一步
- 如果最后fast == slow,则说明该链表存在环形结构;如果最后 fast == null || fast.next == null,则说明该链表不存在环形结构
- 检验链表为null时,代码是否满足
代码如下:
public class Solution {public boolean hasCycle(ListNode head) {ListNode fast = head;ListNode slow = head;while(fast != null && fast.next != null) {fast = fast.next.next;slow = slow.next;if(fast == slow) {return true;}}return false;}
}
2.9 求环的入口点
题目链接:142. 环形链表 II
解题思路:
- 先判断链表结构中是否存在环(在2.8代码中进行略微修改即可)
- 求交点: 让一个引用从链表起始位置开始,一个引用从相遇点位置开始,两个引用每次都走一步,最终相遇时的节点即为交点(原因如下)
数学推导:

代码如下:
public class Solution {public ListNode detectCycle(ListNode head) {ListNode fast = head;ListNode slow = head;while(fast != null && fast.next != null) {fast = fast.next.next;slow = slow.next;//这个if语句必须放在下面,否则该if语句第一次就会成立,//因为fast和slow第一次都是headif(fast == slow){break;}}if(fast == null || fast.next == null) {return null;}slow = head;while(fast != slow) {fast = fast.next;slow = slow.next;}return fast;}
}
2.10 合并两个有序链表
题目链接:21. 合并两个有序链表
解题思路:
- 创建一个带头结点的单链表
- 依次对比两个链表的数值大小,小的尾插到新链表尾部
- 当一个链表被新链表连接完时,另一个链表剩下的部分直接尾插到新链表的尾部
- 检验当一个链表为null时,代码是否满足
代码如下:
class Solution {public ListNode mergeTwoLists(ListNode list1, ListNode list2) {ListNode cur1 = list1;ListNode cur2 = list2;ListNode newHead = new ListNode();//NewHead为带头结点ListNode curN = newHead;while(cur1 != null && cur2 != null) {if(cur1.val < cur2.val) {curN.next = cur1;cur1 = cur1.next;} else {curN.next = cur2;cur2 = cur2.next;}curN = curN.next;}if(cur1 == null) {curN.next = cur2;}if(cur2 == null) {curN.next = cur1;}return newHead.next;}
}
相关文章:
无头单向非循环链表实现 and leetcode刷题
无头单向非循环链表实现 1. 单链表的模拟实现IList.java接口:MySingleList.java文件: 2. leetcode刷题2.1 获取链表的中间节点2.2 删除链表中所有值为value的元素2.3 单链表的逆置2.4 获取链表倒数第k个节点2.5 给定 x, 把一个链表整理成前半部分小于 x,…...
Ubuntu系统上安装Apache和WordPress
** 第一步跟新系统包 ** 首先跟新系统包 sudo apt update sudo apt upgrade第二步下载安装apache sudo apt install apache2 ##查看apache的状态是否启动成功 sudo systemctl status apache2 ##查看服务器的ip地址 sudo ip a通过ip地址进行访问apache页面 第三步下载安装…...
Doze和AppStandby白名单配置方法和说明
机制 配置路径 配置案例 说明 影响机制 调试命令 Doze /platform/frameworks/base /data/etc/platform.xml allow-in-power-save 【系统应用Doze白名单配置】 Doze\Job\AppStandby\Alarm\WakeLock\Sync 查看Doze白名单:adb shell dumpsys deviceidle 添加Doze白名单…...
坑2.Date类型的请求参数
前端 <el-form-item label"结束日期" prop"endTime"><el-date-pickerv-model"dataForm.endTime"type"date"value-format"yyyy-MM-dd HH:mm:ss"placeholder"选择日期"></el-date-picker></el…...
javaweb ajax maven mybatis spring springmvc 在项目中有什么用, 举例说明
JavaWeb是一种基于Java语言的Web开发技术,可以用来开发动态网站和Web应用程序。 AJAX(Asynchronous JavaScript and XML)是一种在Web开发中用于实现异步通信的技术,可以在不刷新整个网页的情况下更新部分页面内容,提升…...
Python编程学习笔记(4)--- 字典
目录 1 什么是字典 2 使用字典 2.1 访问字典中的值 2.2 添加键值对 2.3 创建空字典 2.4 修改字典中的值 2.5 删除键值对 2.6 类似键值对组成的字典 2.7 使用get()来访问值 3 遍历字典 3.1 遍历所有键值对 3.2 遍历字典中的所有键 3.3 按照特…...
会员运营体系设计及SOP梳理
一些做会员的经验和方法分享给大家,包括顶层思考、流程的梳理、组织的建立,后续会做成系列,最近几期主要围绕顶层策略方面,以下是核心内容的整理: 1、会员运营体系设计 顶层设计与关键业务定位:建立客户运营…...
SQL 自定义函数
概念 自定义函数是用户根据自己的业务逻辑或计算需求创建的函数。这些函数可以接收一个或多个输入参数,执行一系列的操作(如计算、数据处理、逻辑判断等),并最终返回一个值或结果集。自定义函数可以被多次重用,提高了…...
C# 下sendmessage和postmessage的区别详解与示例
文章目录 1、SendMessage2、PostMessage3、两者的区别: 总结 在C#中,SendMessage和PostMessage是两个用于Windows编程的API,它们用于向窗口发送消息。这两个方法都位于System.Windows.Forms命名空间中,通常用于自动化Windows应用程…...
Transformer重要论文与书籍 - Transformer教程
近年来,人工智能领域中的Transformer模型无疑成为了炙手可热的研究对象。从自然语言处理(NLP)到计算机视觉,Transformer展现出了前所未有的强大能力。今天,我们将探讨Tra在当今的人工智能和机器学习领域,Tr…...
android13 rom 开发总纲说明
1. 这里是文章总纲,可以在这里快速找到需要的文章。 2. 文章一般是基于标准的android13,有一些文章可能会涉及到具体平台,例如全志,瑞芯微等一些平台。 3.系统应用 3.1系统应用Launcher3桌面相关: 3.2系统应用设置S…...
2.线性回归
简化的房价模型 假设1:影响房价的关键因素时卧室个数,卫生间和居住面积,记为 x 1 , x 2 , x 3 x_1,x_2,x_3 x1,x2,x3 假设2:成交价时关键因素的加权和: y w 1 x 1 w 2 x 2 w 3 x 3 b y w_1x_1w_2x_2w_3x…...
一文了解java中Optional
文章目录 1. Optional简介2. 常用的接口2.1 常用接口简单使用2.1.1 创建的常用方法2.1.2 获取值的常用方法2.1.3 判定的常用方法2.1.4 判定后的操作方法2.2 map方法介绍 2.2 其他方法2.2.1 Filter 方法2.2.2 FlatMap 方法 3. 常用的实例4. 总结 1. Optional简介 Optional是在ja…...
提示词工程(Prompt Engineering)是什么?
一、定义 Prompt Engineering 提示词工程(Prompt Engineering)是一项通过优化提示词(Prompt)和生成策略,从而获得更好的模型返回结果的工程技术。 二、System message 系统指令 System message可以被广泛应用在&am…...
vue对axios进行请求响应封装
一、原因 像是在一些业务逻辑上,比如需要在请求之前展示loading效果,或者在登录的时候判断身份信息(token)等信息有没有过期,再者根据服务器响应回来的code码进行相应的提示信息。等等在请求之前,之后做的一…...
快速测试electron环境是否安装成功
快速测试electron环境是否安装成功 测试代码正确运行的效果运行错误的效果v22.4.1 版本无法使用v20.15.1版本无法使用v18.20.4 版本无法使用 终极解决办法 测试代码 1.npx create-electron-app my-electron-app 2.cd my-electron-app 3.npm start 正确运行的效果 环境没问题…...
数电设计提问求帮助,出租车计费器。
🏆本文收录于《CSDN问答解惑-》专栏,主要记录项目实战过程中的Bug之前因后果及提供真实有效的解决方案,希望能够助你一臂之力,帮你早日登顶实现财富自由🚀;同时,欢迎大家关注&&收藏&…...
xcode项目添加README.md文件并进行编辑
想要给xcode项目添加README.md文件其实还是比较简单的,但是对于不熟悉xcode这个工具的人来讲,还是有些陌生,下面简单给大家讲一下流程。 选择“文件”>“新建”>“文件”,在其他(滚动到工作表底部)下…...
基于 cookiecutter 的 python 项目模板
Cookiecutter 介绍 使用 Python 这种动态语言进行 web 开发,团队中经常会遇到的问题就是代码的质量比较难控制。Python 语言本身灵活性比较高,不加控制的情况下代码质量可能最后很难维护。而且代码的各方面的标准,比如提示的 lint࿰…...
如何玩转澳大利亚Facebook直播?
近年来,直播带货已经成为国内最赚钱的行业之一,各种玩法也越来越成熟。然而,在海外市场,尤其是澳大利亚,直播带货仍然是一片蓝海。作为社交媒体营销的主阵地,Facebook的直播功能却常常被卖家忽视。那么&…...
R语言AI模型部署方案:精准离线运行详解
R语言AI模型部署方案:精准离线运行详解 一、项目概述 本文将构建一个完整的R语言AI部署解决方案,实现鸢尾花分类模型的训练、保存、离线部署和预测功能。核心特点: 100%离线运行能力自包含环境依赖生产级错误处理跨平台兼容性模型版本管理# 文件结构说明 Iris_AI_Deployme…...
蓝桥杯 2024 15届国赛 A组 儿童节快乐
P10576 [蓝桥杯 2024 国 A] 儿童节快乐 题目描述 五彩斑斓的气球在蓝天下悠然飘荡,轻快的音乐在耳边持续回荡,小朋友们手牵着手一同畅快欢笑。在这样一片安乐祥和的氛围下,六一来了。 今天是六一儿童节,小蓝老师为了让大家在节…...
智能在线客服平台:数字化时代企业连接用户的 AI 中枢
随着互联网技术的飞速发展,消费者期望能够随时随地与企业进行交流。在线客服平台作为连接企业与客户的重要桥梁,不仅优化了客户体验,还提升了企业的服务效率和市场竞争力。本文将探讨在线客服平台的重要性、技术进展、实际应用,并…...
【SQL学习笔记1】增删改查+多表连接全解析(内附SQL免费在线练习工具)
可以使用Sqliteviz这个网站免费编写sql语句,它能够让用户直接在浏览器内练习SQL的语法,不需要安装任何软件。 链接如下: sqliteviz 注意: 在转写SQL语法时,关键字之间有一个特定的顺序,这个顺序会影响到…...
涂鸦T5AI手搓语音、emoji、otto机器人从入门到实战
“🤖手搓TuyaAI语音指令 😍秒变表情包大师,让萌系Otto机器人🔥玩出智能新花样!开整!” 🤖 Otto机器人 → 直接点明主体 手搓TuyaAI语音 → 强调 自主编程/自定义 语音控制(TuyaAI…...
Device Mapper 机制
Device Mapper 机制详解 Device Mapper(简称 DM)是 Linux 内核中的一套通用块设备映射框架,为 LVM、加密磁盘、RAID 等提供底层支持。本文将详细介绍 Device Mapper 的原理、实现、内核配置、常用工具、操作测试流程,并配以详细的…...
Python 包管理器 uv 介绍
Python 包管理器 uv 全面介绍 uv 是由 Astral(热门工具 Ruff 的开发者)推出的下一代高性能 Python 包管理器和构建工具,用 Rust 编写。它旨在解决传统工具(如 pip、virtualenv、pip-tools)的性能瓶颈,同时…...
NPOI Excel用OLE对象的形式插入文件附件以及插入图片
static void Main(string[] args) {XlsWithObjData();Console.WriteLine("输出完成"); }static void XlsWithObjData() {// 创建工作簿和单元格,只有HSSFWorkbook,XSSFWorkbook不可以HSSFWorkbook workbook new HSSFWorkbook();HSSFSheet sheet (HSSFSheet)workboo…...
uniapp 字符包含的相关方法
在uniapp中,如果你想检查一个字符串是否包含另一个子字符串,你可以使用JavaScript中的includes()方法或者indexOf()方法。这两种方法都可以达到目的,但它们在处理方式和返回值上有所不同。 使用includes()方法 includes()方法用于判断一个字…...
深度剖析 DeepSeek 开源模型部署与应用:策略、权衡与未来走向
在人工智能技术呈指数级发展的当下,大模型已然成为推动各行业变革的核心驱动力。DeepSeek 开源模型以其卓越的性能和灵活的开源特性,吸引了众多企业与开发者的目光。如何高效且合理地部署与运用 DeepSeek 模型,成为释放其巨大潜力的关键所在&…...
