数据结构奇妙旅程之顺序表和链表

꒰˃͈꒵˂͈꒱ write in front ꒰˃͈꒵˂͈꒱
ʕ̯•͡˔•̯᷅ʔ大家好,我是xiaoxie.希望你看完之后,有不足之处请多多谅解,让我们一起共同进步૮₍❀ᴗ͈ . ᴗ͈ აxiaoxieʕ̯•͡˔•̯᷅ʔ—CSDN博客
本文由xiaoxieʕ̯•͡˔•̯᷅ʔ 原创 CSDN 如需转载还请通知˶⍤⃝˶
个人主页:xiaoxieʕ̯•͡˔•̯᷅ʔ—CSDN博客
系列专栏:xiaoxie的JAVA系列专栏——CSDN博客●'ᴗ'σσணღ*
我的目标:"团团等我💪( ◡̀_◡́ ҂)" 

( ⸝⸝⸝›ᴥ‹⸝⸝⸝ )欢迎各位→点赞👍 + 收藏⭐️ + 留言📝​+关注(互三必回)!

目录

​编辑一.顺序表

1.底层实现

2.构造方法

3.常用方法

4.Arrays的遍历方法

​编辑5.实战演示

一.顺序表

1.底层实现

首先我们要清楚,数据结构是一门逻辑十分清晰的学科,所以我们需要清楚的认识到每个结构的底层是什么,这样才能更好的掌握这个结构。

顺序表是用一段 物理地址连续 的存储单元依次存储数据元素的线性结构,一般情况下采用数组存储。在数组上完成 数据的增删查改。

2.构造方法

在集合框架中, ArrayList 是一个普通的类,实现了 List接口,并且ArrayList是一个泛型类
而ArrayList的构造方法如下所示
public class Text {public static void main(String[] args) {//无参List<Integer> list = new ArrayList<>();//指定顺序表初始容量List<Integer> list1 = new ArrayList<>(20);list1.add(1);list1.add(11);//ArrayList(Collection<? extends E> c) 利用其他 Collection 构建 ArrayListList<Integer> list2 = new ArrayList<>(list1);}
}

3.常用方法

ArrayList是一个普通的类,实现了List接口,所以它实现了许多接口内的方法,博主现在为大家列举出一些常用的方法

List<Integer> list = new ArrayList<>();//尾插入元素list.add(1);list.add(12);list.add(11);//指定位置插入参数为 位置 和 元素list.add(2,3);//顺序表长度list.size();//删除指定位置元素list.remove(2);//是否包含某个元素list.contains(4);//截取部分 listlist.subList(0,1);//获取下标 index 位置元素list.get(2);//将下标 index 位置元素设置为 elementlist.set(3,9);//返回第一个 o 所在下标list.indexOf(8);//返回最后一个 o 的下标list.lastIndexOf(8);

4.Arrays的遍历方法

1.直接输出

 List<Integer> list = new ArrayList<>();//尾插入元素list.add(1);list.add(12);list.add(11);System.out.println(list);

2.for循环和for-each

List<Integer> list = new ArrayList<>();//尾插入元素list.add(1);list.add(12);list.add(11);System.out.println("for循环遍历");for (int i = 0; i < list.size(); i++) {System.out.print(list.get(i)+" ");}System.out.println();System.out.println("for-each循环遍历");for (Integer x:list) {System.out.print(x+" ");}

3.使用迭代器

List<Integer> list = new ArrayList<>();//尾插入元素list.add(1);list.add(12);list.add(11);System.out.println("迭代器正序输出");Iterator<Integer>it = list.iterator();while (it.hasNext()) {System.out.print(it.next()+" ");}System.out.println();System.out.println("迭代器逆序输出");ListIterator <Integer> it1 = list.listIterator(list.size());while (it1.hasPrevious()) {System.out.print(it1.previous()+" ");}}

5.实战演示

杨辉三角 力扣的118题,难度不大,感兴趣的朋友可以去尝试一下

解答代码如下

/*11 1 1 2 11 3 3 11 4 6 4 1 由图可知每一行的第一个为1,最后一个也为一中间部分等于上一行的前一个加上上一行同一列的一个,并且最后一个也为一。
*/
class Solution {public List<List<Integer>> generate(int numRows) {List<Integer> list = new ArrayList<>();List<List<Integer>> ans = new ArrayList<>();list.add(1);//第一行ans.add(list);for(int i = 1;i < numRows;i++) {List<Integer> temp = new ArrayList<>();temp.add(1);//每一行的第一个都为一List<Integer> row =ans.get(i-1);//上一行for(int j = 1; j <i;j++) {int x = row.get(j)+row.get(j-1);//中间部分为上一行的前一个加上上一行同一列的一个temp.add(x);}temp.add(1);//一行的最后一个元素也为1ans.add(temp);}return ans;}
}

以上是博主解答这题的思路和代码,时间复杂度为O(n^2)如果你有更好的方法或者是不懂的地方都可以私聊博主,大家一起交流进步

二.链表

1.底层实现

链表是一种 物理存储结构上非连续 存储结构,数据元素的 逻辑顺序 是通过链表中的 引用链接 次序实现的,它的结构如下
 

当然链表的结构不仅仅是这一种其他还有1. 单向或者双向  2. 带头或者不带头  3. 循环或者非循环

虽然有这么多的链表的结构,但是我们重点掌握两种 :
无头单向非循环链表 结构简单 ,一般不会单独用来存数据。实际中更多是作为 其他数据结构的子结构 ,如
哈希桶、图的邻接表等等。另外这种结构在 笔试面试 中出现很多。
无头双向链表 :在 Java 的集合框架库中 LinkedList 底层实现就是无头双向循环链表。

2.无头单向非循环链表

无头单向非循环链表是一种数据结构,其特点是没有头节点、只有一个指向第一个节点的指针,而且最后一个节点的指针为空(NULL),形成一个单向链表的尾部。这意味着如果需要查找第一个节点,必须从指向第一个节点的指针开始遍历链表。此外,非循环链表是一个单向链表,从第一个节点开始一直到最后一个节点,而最后一个节点指向空(NULL),这样就可以避免链表中的死循环问题。

3.无头双向非循环链表

无头双向非循环链表是一种数据结构,它由多个节点组成,每个节点有两个指针,一个指向前一个节点,一个指向后一个节点。和单向链表不同的是,双向链表可以向前和向后遍历,而且插入和删除节点的操作比较方便。

无头表示链表没有头节点,也就是第一个节点就是链表的起始节点。非循环表示链表没有环,也就是最后一个节点的指针指向 NULL。

在无头双向非循环链表中,我们可以从任意一个节点开始遍历,也可以逆序遍历整个链表。双向链表比单向链表多了一个指针,所以它在一些场景下比单向链表更加方便,比如需要频繁地插入和删除节点时。但是,它也需要更多的内存空间来存储指针。

4.顺序表和链表之间的区别

    顺序表和链表是两种数据结构,它们的主要区别在于存储方式和操作效率不同:

  1. 存储方式不同:顺序表使用一组连续的存储单元存储数据,而链表则使用指针将一组零散的存储单元串联起来。

  2. 随机访问效率不同:顺序表通过下标可以直接访问任意位置的元素,时间复杂度为O(1),但插入和删除操作需要移动数据,时间复杂度为O(n)。而链表不能通过下标直接访问元素,插入和删除操作只需要修改指针,时间复杂度为O(1),但是随机访问需要遍历整个链表,时间复杂度为O(n)。

  3. 空间利用率不同:顺序表预先分配存储空间大小固定,当存储空间不够时需要重新分配内存,可能浪费内存。而链表通过动态分配内存,只在需要时才分配所需的存储空间,空间利用率较高。

总之,应根据实际需求选择合适的数据结构,如果需要频繁随机访问,可以选择顺序表;如果需要频繁插入和删除,可以选择链表。

5.链表面试题

1.给定一个链表,返回链表开始入环的第一个节点。 如果链表无环,则返回 NULL 链接

public class Solution {// 定义一个方法detectCycle,输入参数为链表头节点head,返回值类型为ListNode(链表节点)public ListNode detectCycle(ListNode head) {// 如果链表为空或只有一个元素,则没有环,直接返回nullif(head == null || head.next == null) {return null;}// 初始化两个指针fast和slow,都指向链表头节点ListNode fast = head;ListNode slow = head;// 使用快慢指针进行遍历while(fast != null && fast.next != null) {// 慢指针每次前进一步slow = slow.next;// 快指针每次前进两步fast = fast.next.next;// 如果快慢指针相遇,则说明存在环if(slow == fast) {break;}}// 如果快指针遍历到链表末尾,说明不存在环,返回nullif(fast == null || fast.next == null) {return null;}// 重新设置快指针fast到链表头节点,慢指针slow保持在相遇点fast = head;// 当快慢指针再次相遇时,相遇点即为环的起点while(slow != fast) {slow = slow.next;fast = fast.next;}// 返回环的起点return fast;}
}

2.链表的回文结构链接

public class PalindromeList {// 定义一个方法chkPalindrome,输入参数为链表头节点head,返回值类型为布尔值(true表示是回文链表,false表示不是)public boolean chkPalindrome(ListNode head) {// 如果链表为空,则直接返回true(空链表是回文链表)if(head == null ) {return true;}// 使用快慢指针找到链表的中间结点ListNode fast = head;ListNode slow = head;while(fast != null && fast.next != null) {fast = fast.next.next; // 快指针每次前进一步slow = slow.next;      // 慢指针每次前进一步}// 翻转链表后半部分ListNode cur = slow.next;while(cur != null) {ListNode curnext = cur.next;cur.next = slow;slow = cur;cur = curnext;}// 判断翻转后的链表与原链表前半部分是否相等while(head != slow) {if(head.val != slow.val) { // 如果两个节点的值不相等,则说明不是回文链表return false;}if(head.next != slow) {   // 奇数情况下,已经比较完所有节点,可以提前结束循环return true;}head = head.next;          // 头节点向后移动一位slow = slow.next;          // 反转后的尾节点向前移动一位}// 遍历结束后没有发现不相等的节点,说明是回文链表return true;}
}

3.将两个有序链表合并为一个新的有序链表并返回。新链表是通过拼接给定的两个链表的所有节点组成的。链接


public class Partition {// 定义一个方法partition,输入参数为链表头节点head和整数x,返回值类型为ListNode(链表节点)public ListNode partition(ListNode head, int x) {// 如果链表为空,则直接返回nullif(head == null) {return null;}// 初始化两个指针de和dx,用于记录小于x的元素链表的尾节点和当前节点ListNode de = null;ListNode dx = null;// 初始化两个指针pe和px,用于记录大于等于x的元素链表的尾节点和当前节点ListNode pe = null;ListNode px = null;// 初始化一个指针cur,指向链表的头节点ListNode cur = head;// 遍历链表中的每个节点while(cur != null) {// 如果当前节点的值小于xif(cur.val < x) {// 如果小于x的元素链表尚未创建,则将当前节点设置为头节点和尾节点if(de == null) {de = cur;dx = cur;}else { // 否则将当前节点添加到小于x的元素链表的末尾dx.next = cur;dx = dx.next;}}else { // 如果当前节点的值大于等于x// 如果大于等于x的元素链表尚未创建,则将当前节点设置为头节点和尾节点if(pe == null) {pe = cur;px = cur;}else { // 否则将当前节点添加到大于等于x的元素链表的末尾px.next = cur;px = px.next;}}// 指针cur向后移动一位cur = cur.next;}// 如果不存在小于x的元素,则直接返回大于等于x的元素链表if(dx == null) {return pe;}// 将大于等于x的元素链表接到小于x的元素链表的末尾dx.next = pe;// 如果存在大于等于x的元素,则将其尾节点的next指针置为nullif(pe != null) {px.next = null;}// 返回小于x的元素链表的头节点return de;}
}

三.结语

数据结构一门逻辑十分严谨的一门学科,而顺序表和链表都是十分基础的结构,也是后续学习的基础,所以应该要多花时间去搞清楚它的底层,好了本文到这里就结束了,感谢您的阅读,祝您一天愉快

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

C++-内联函数

目录 一.什么是内联函数 1.内联函数的概念 2.内联函数的定义 二.C中引入内联函数的原因 三.什么样的函数适合被声明为内联呢&#xff1f; 四.面试题 一.什么是内联函数 1.内联函数的概念 以inline修饰的函数叫做内联函数&#xff0c;编译时C编译器会在调用内联函数的地方展开…

springboot 2.4.4集成 hikari连接池多数据源实例

文章目录 前言一、配置步骤1.1 pom配置1.2 application.properties配置1.3 DataSourceContextHolder类1.4 DynamicDataSource1.5 DataSourceconfig类配置1.6 配置TargetDataSource注解1.7 切面方法1.8 dao的写法 二、测试验证2.1 启动springboot项目2.2 检查数据库连接2.3 debu…

python——进程常用功能

Python的multiprocessing模块提供了强大的并行处理能力&#xff0c;以下是几个功能的详细解释&#xff1a; join(): 在multiprocessing中&#xff0c;join方法用于阻塞主进程直到指定的进程终止。这对于确保所有子进程在程序结束前完成其工作是很有用的。deamon(): 在multipro…

基于51单片机的十字路口交通灯_5s黄灯倒计时闪烁

基于51单片机十字路口交通灯_5s黄灯闪烁 &#xff08;程序仿真仿真视频&#xff09; 仿真&#xff1a;proteus 7.8 程序编译器&#xff1a;keil 4/keil 5 编程语言&#xff1a;C语言 设计编号&#xff1a;J006 功能要求 交通灯运行状态&#xff1a; &#xff08;1&…

硬件基础:二极管

基本定义 二极管的内部其实就是一个PN结。 把PN结封装起来&#xff0c;两边加上两个电极&#xff0c;就组成了半导体二极管。简称二极管&#xff08;Diode&#xff09; 二极管和PN结一样&#xff0c;具有单向导通性&#xff1a; 外观和正负极 常见芯片封装如下&#xff1a; 一般…

java 工具类: CompareUtils(比较对象字段值变化)

一、前言 我们在工作中&#xff0c;可能会在日志中记录数据的变化情况或者在公共处理的数据增加一个日志页面&#xff0c;记录每次修改的变化。我们可以根据CompareUtils工具类比较数据前后发生了怎样的变化, 这样我们就可以知道数据做了哪些改变. 二、条件限制 在写这个通用…

ImportError: cannot import name ‘metadata‘ from ‘importlib‘

yolov8 编译问题 ImportError: cannot import name ‘metadata’ from ‘importlib’ 将 from importlib import metadata 更改为 import importlib_metadata as metadata

《明解C语言》第三版 (入门篇) 第十三章练习答案

练习13-1 //打开与关闭文件#include <stdio.h>int main() {FILE* fd;char str[128];printf("请输入你要打开的文件\n");scanf("%s", str);fd fopen("*str","r");//只读方式打开if (fd NULL){puts("无法打开abc这个文件&…

【WinForm.NET开发】创建 Windows 窗体应用

本文内容 创建项目创建应用程序运行应用程序 本文演示创建一个具有基于 Windows 的用户界面 (UI) 的简单 C# 应用程序。 1、创建项目 首先&#xff0c;创建 C# 应用程序项目。 项目类型随附了所需的全部模板文件&#xff0c;无需添加任何内容。 打开 Visual Studio。在“开…

vs 安装 qt qt扩展

1 安装qt 社区版 免费 Download Qt OSS: Get Qt Online Installer 2 vs安装 qt vs tools 3 vs添加 qt添加 bin/cmake.exe 路径 3.1 扩展 -> qt versions 3.2

【设计模式-4.1】行为型——观察者模式

说明&#xff1a;本文介绍设计模式中行为型设计模式中的&#xff0c;观察者模式&#xff1b; 商家与顾客 观察者模式属于行为型设计模式&#xff0c;关注对象的行为。以商家与顾客为例&#xff0c;商家有商品&#xff0c;顾客来购买商品&#xff0c;如果商家商品卖完了&#…

Vue3中teleport如何使用

Vue 3作为一种流行的JavaScript框架&#xff0c;一直以来都在努力提供更便捷、灵活的开发体验。本文将深入解析Teleport&#xff0c;包括其详细的使用方法、源码实现机制以及在实际项目中的应用场景。 一、Teleport是什么&#xff1f; Vue 3中的Teleport允许开发者将组件的内容…

Vue+ElementUI+C#前后端分离:监控长耗时任务的实践

想象一下&#xff0c;我们正在构建一个Web应用&#xff0c;需要实现一个数据报告的导出功能。这听起来很简单&#xff0c;不是吗&#xff1f;但是&#xff0c;随着深入开发&#xff0c;我们意识到导出过程比预期的要复杂和耗时得多。由于报告的数据量巨大&#xff0c;后端需要花…

PostgreSQL有意思的现象:支持不带列的表

1、前言 以前从没有试过建一张表&#xff0c;不带任何列。在PG中却支持这种语法。这是个什么鬼? 最近&#xff0c;把PG源码扒了下&#xff0c;简单浏览了下最近的一些merge。其中有一个fix&#xff1a; eeb0ebad79 ("Fix the initial sync tables with no columns.&qu…

〖大前端 - 基础入门三大核心之JS篇㊺〗- 定时器和延时器

说明&#xff1a;该文属于 大前端全栈架构白宝书专栏&#xff0c;目前阶段免费&#xff0c;如需要项目实战或者是体系化资源&#xff0c;文末名片加V&#xff01;作者&#xff1a;不渴望力量的哈士奇(哈哥)&#xff0c;十余年工作经验, 从事过全栈研发、产品经理等工作&#xf…

LeetCode 每日一题 2023/11/27-2023/12/3

记录了初步解题思路 以及本地实现代码&#xff1b;并不一定为最优 也希望大家能一起探讨 一起进步 目录 11/27 907. 子数组的最小值之和11/28 1670. 设计前中后队列11/29 2336. 无限集中的最小数字11/30 1657. 确定两个字符串是否接近12/1 2661. 找出叠涂元素12/2 1094. 拼车12…

本地缓存和分布式缓存

一、引言 在当今的大数据时代&#xff0c;数据缓存已成为提升应用性能和效率的重要策略。缓存能够降低数据访问延迟&#xff0c;提高系统响应速度&#xff0c;从而改善用户体验。根据存储位置和应用场景的不同&#xff0c;缓存技术分为本地缓存和分布式缓存两种。本文将详细介绍…

【Rust日报】2023-12-01 KCL v0.7 版本发布

RFC&#xff1a;在选择依赖项时使 Cargo 遵循最低支持的 Rust 版本 (MSRV) 概括内容是&#xff0c;为需要使用旧版本 Rust 的开发者提供了一条快乐之路&#xff0c;具体方法是&#xff1a; 在 Cargo 解析依赖关系时&#xff0c;优先选择与 MSRV&#xff08;最低支持的 Rust 版本…

如何使用PHPUnit编写一个PHP单元测试-简单的代码示例

在软件开发过程中&#xff0c;单元测试是一种重要的测试方法&#xff0c;可以确保代码的质量和可靠性。在PHP开发中&#xff0c;也可以通过编写单元测试来验证代码的正确性。下面将介绍一些编写PHP单元测试的基本步骤和常用工具。 首先&#xff0c;你需要一个PHP单元测试框架&…

3D场景建模工具

在线工具推荐&#xff1a; 三维数字孪生场景工具 - GLTF/GLB在线编辑器 - Three.js AI自动纹理化开发 - YOLO 虚幻合成数据生成器 - 3D模型在线转换 - 3D模型预览图生成服务 1. 什么是3D场景建模&#xff1f; 3D场景建模是一种通过计算机图形学技术&#xff0c;将现实世…