- ubuntu12.04环境下使用kvm ioctl接口实现最简单的虚拟机
- Ubuntu 通过无线网络安装Ubuntu Server启动系统后连接无线网络的方法
- 在Ubuntu上搭建网桥的方法
- ubuntu 虚拟机上网方式及相关配置详解
CFSDN坚持开源创造价值,我们致力于搭建一个资源共享平台,让每一个IT人在这里找到属于你的精彩世界.
这篇CFSDN的博客文章一篇文章带你玩转JAVA单链表由作者收集整理,如果你对这篇文章有兴趣,记得点赞哟.
。
。
链表是一种物理存储结构上非连续的存储结构,数据元素的逻辑顺序是通过链表中的引用链接次序实现的 。
上章介绍到顺序表适合用作查询和修改,而不适合用作插入和删除。并且它增容时容易造成空间浪费。而链表则具有以下的特点 。
适合用作插入和删除随用随取,避免了空间的浪费不适合用作查询和修改 。
。
链表其实可以想象成一条被打了一些结的绳子 。
而实际上,链表就是由一个个节点构成的,只不过每个节点一般有两个域 。
数值域 data: 里面存储数据 。
next 域: 里面存储的是下一个节点的地址(是引用变量) 。
其中 。
尾节点: 当这个节点的 next 域为 null 时,该节点就是尾节点 。
头节点: 整个链表的第一个节点就是头节点 。
实际中的链表结构非常多,通过以下的情况的组合可以有8种链表的结构(比如带头单向循环链表就是一种) 。
单向、双向 。
带头节点、不带头 。
节点循环、非循环 。
而上图所示,就是一个单向不带头非循环链表.
可能有人会疑问,上述图片中不是存在头节点吗?那为啥它又是一个不带头节点的链表呢?我将上述图片实例稍稍改一下就是带头节点的了 。
我在原有的头节点前面又多了一个节点,并且这个节点的数值域里面的数据可有可无,因为对其没有影响。而这里面的头节点只是一个标志,标识这个节点是该链表的头 。
那带头节点的和不带头结点的链表有啥差别呢?
不带头: 这个链表的头节点可能发生改 。
变带头: 这个链表的头节点不会发生改变 。
那循环链表又是啥样的呢?我们可以将上述单向不带头非循环链表稍微修改一下 。
即将尾节点的 next 域存储了头节点的地址 。
我们知道一个节点一般就两个域,但如果是双向链表,则就有三个域了 。
数值域 data: 里面存储数据 。
next 域: 里面存储的是下一个节点的地址(是引用变量) 。
prev 域: 里面存储的是上一个节点的地址(是引用变量) 。
我们画一个不带头双向非循环链表就是这样的 。
接下来我将主要介绍单向不带头非循环链表和双向不带头非循环链表,这两种链表也是经常出现在面试题中的 。
。
。
上述通过图解,大家应该对单向不带头非循环链表应该有了了解。那么代码中该怎么实现呢?
我们知道,链表应该可以看作一个类,而节点本身其实也可以抽象的看作成一个类 。
首先对于节点类,代码可以写出这样 。
class Node{ public int val; public Node next; public Node(int val){ this.val=val; }}
先定义了数值域,再定义了 next 域(由于 next 存储的是下一个节点的地址,故写出上述那样)。而我们初始时可以知道 val,所以可以写个构造函数对 val 进行初始化,而该节点的 next 域初始时是不可能知道的,所以不对 next 做初始化 。
如果创造一个 Node 对象就可以写出 。
Node node = new Node(10),
即头节点的数值域为10.
再定义链表类,代码可以写成这样 。
public class MyLinkedList { public Node head;}
我们定义了一个头节点,这是我们很容易发现的,就比如我要对该链表进行头插,则该链表的头节点一直都在改变。所以写上述的代码就是为了标识单链表的头节点.
。
接下来我们来实现单向不带头非循环链表的一些操作 。
打印链表 。
public void display(){ Node cur =this.head; while(cur!=null){ System.out.print(cur.val + " "); cur=cur.next; } System.out.println();}
求链表的长度 。
public int size(){ Node cur=this.head; int count=0; while(cur!=null){ count++; cur=cur.next; } return count;}
查找值 key 是否在单链表中 。
public boolean contains(int key){ Node cur=this.head; while(cur!=null){ if(key==cur.val){ return true; } cur=cur.next; } return false;}
清除单链表 。
public void clear(){ this.head.val=0; this.head.next=null;}
头插法 。
public void addFirst(int data){ Node node=new Node(data); if(head==null){ this.head=node; }else { node.next = this.head; this.head = node; }}
尾插法 。
public void addLast(int data){ Node node = new Node(data); if(this.head==null){ this.head=node; }else { Node cur = this.head; while (cur.next != null) { cur = cur.next; } cur.next = node; }}
通过下标找到前驱节点(第一个节点下标为0) 。
public Node searchPrev(int index){ Node cur = this.head; int count=0; while(count!=index-1){ cur=cur.next; count++; } return cur;}
任意位置插入 。
public void addIndex(int index, int data){ if(index<0 || index>size()){ throw new RuntimeException("index 不合法"); } if(index==0){ addFirst(data); return; } if(index==size()) { addLast(data); return; } Node node =new Node(data); Node cur =searchPrev(index); node.next=cur.next; cur.next=node;}
通过值 key 找到前驱节点(第一个节点下标为0) 。
public Node searchPrevNode(int key){ Node cur = this.head; while(cur.next!=null){ if(cur.next.val==key) { return cur; } cur=cur.next; } return null;}
删除第一次出现的值为 key 的节点 。
public void remove(int key){ if(this.head==null){ return; } if(key==this.head.val){ this.head=this.head.next; return; } Node node = searchPrevNode(key); if(node==null){ System.out.println("链表中无要删除的元素"); }else { node.next = node.next.next; }}
删除所有出现的值为 key 的节点 。
public void removeAllKey(int key){ if(this.head==null){ return; } Node prev=this.head; Node cur=this.head.next; while(cur!=null){ if(cur.val==key){ cur=cur.next; prev.next=cur; }else{ prev=cur; cur=cur.next; } } if(this.head.val==key){ this.head=this.head.next; }}
呼,写的我人傻了 。
你以为结束了吗?NO!下面还有一大波链表面试题等着我和你,冲吧,牛牛! 。
。
删除链表中等于给定值 val 的所有节点 OJ 链接 。
反转一个单链表 OJ 链接 。
给定一个带有头结点 head 的非空单链表,返回链表的中间结点。如果有两个中间结点,则返回第二个中间结点 OJ 链接 。
输入一个链表,输出该链表中倒数第k个结点 OJ 链接 。
将两个有序链表合并为一个新的有序链表并返回。新链表是通过拼接给定的两个链表的所有节点组成的 OJ 链接 。
编写代码,以给定值x为基准将链表分割成两部分,所有小于x的结点排在大于或等于x的结点之前 OJ 链接 。
在一个排序的链表中,存在重复的结点,请删除该链表中重复的结点,重复的结点不保留,返回链表头指针 OJ 链接 。
链表的回文结构 OJ 链接 。
输入两个链表,找出它们的第一个公共结点 OJ 链接 。
给定一个链表,判断链表中是否有环 OJ 链接 。
给定一个链表,返回链表开始入环的第一个节点。 如果链表无环,则返回 null OJ 链接 。
如果你已经可以轻松应对上述题目了,可以继续去下面的链接中做题 Leetcode OJ 链接 和 牛客 OJ 链接 。
。
今天简单了解了链表,以及对单向不带头非循环链表做了一些基操的实现。下章将包含个人的上述前11道 OJ 题的题解,以及了解双向不带头非循环链表 。
本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注我的更多内容! 。
原文链接:https://blog.csdn.net/weixin_51367845/article/details/120019451 。
最后此篇关于一篇文章带你玩转JAVA单链表的文章就讲到这里了,如果你想了解更多关于一篇文章带你玩转JAVA单链表的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
单向链表 单向链表比顺序结构的线性表最大的好处就是不用保证存放的位置,它只需要用指针去指向下一个元素就能搞定。 单链表图解 图画的比较粗糙,简单的讲解一下: 上面四个长方形,每个长方
使用TCP,我正在设计一些类似于next的程序。 客户端在许多线程中的接收正在等待一台服务器的发送消息。但是,这是有条件的。 recv正在等待特定的发送消息。 例如 客户 thread 1: recv
我正在编写正则表达式来验证电子邮件。唯一让我困惑的是: 顶级域名可以使用单个字符吗?(例如:lockevn.c) 背景:我知道顶级域名可以是 2 个字符到任意字符(.uk、.us 到 .canon、.
是否可以在单个定义中定义同一 Controller 的多个路由? 例如: 我想要一个单一的定义 /, /about, /privacy-policy 使用类似的东西 _home: pat
我正在使用 objective-c开发针对 11.4 iOS 的单 View 应用程序,以及 Xcode版本是 9.4.1。 创建后有Main.storyboard和LaunchScreen.stor
我一直在尝试在 shell 程序中实现管道结构,如果我执行简单的命令(例如“hello | rev”),它就可以工作 但是当我尝试执行“head -c 1000000/dev/urandom | wc
此表包含主机和接口(interface)列UNIQUE 组合* 编辑:这个表也有一个自动递增的唯一 ID,抱歉我应该在之前提到这个 ** | host.... | interface..... |
我想将具有固定补丁大小的“std filter”应用于单 channel 图像。 也就是说,我希望 out[i,j] 等于 img[i,j] 附近的像素值的标准值。 对于那些熟悉 Matlab 的人,
假设我想进行网络调用并使用 rx.Single,因为我希望只有一个值。 我如何应用replay().autoConnect() 这样的东西,这样当我从多个来源订阅时网络调用就不会发生多次?我应该使用
我将图像从 rgb 转换为 YUV。现在我想单独找到亮度 channel 的平均值。你能告诉我如何实现这一目标吗?此外,有没有办法确定图像由多少个 channel 组成? 最佳答案 你可以这样做: #
在比较Go和Scala的语句结束检测时,我发现Scala的规则更丰富,即: A line ending is treated as a semicolon unless one of the foll
在IEEE 1800-2005或更高版本中,&和&&二进制运算符有什么区别?它们相等吗? 我注意到,当a和b的类型为bit时,这些coverpoint定义的行为相同: cp: coverpoint a
我正在使用Flutter的provider软件包。我要实现的是为一个 View 或页面提供一个简单的提供程序。因此,我在小部件中尝试了以下操作: Widget build(BuildContext c
我正在尝试在 cython 中使用 openmp。我需要在 cython 中做两件事: i) 在我的 cython 代码中使用 #pragma omp single{} 作用域。 ii) 使用#pra
我正在尝试从转义字符字符串中删除单引号和双引号。它对单引号 ' 或双自动 " 不起作用。 请问有人可以帮忙吗? var mysting = escapedStr.replace(/^%22/g, '
我正在尝试在 cython 中使用 openmp。我需要在 cython 中做两件事: i) 在我的 cython 代码中使用 #pragma omp single{} 作用域。 ii) 使用#pra
我正在使用 ANT+ 协议(protocol),将智能手机与 ANT+ USB 加密狗连接,该加密狗通过 SimulANT+ 连接到 PC。 SimulANT+ 正在模拟一个心率传感器,它将数据发送到
有人可以解释/理解单/多线程模式下计算结果的不同吗? 这是一个大约的例子。圆周率的计算: #include #include #include const int itera(100000000
我编写了一个粗略的阴影映射实现,它使用 6 个不同的 View 矩阵渲染场景 6 次以创建立方体贴图。 作为优化,我正在尝试使用几何着色器升级到单 channel 方法,但很难从我的着色器获得任何输出
尝试使用 Single-Spa 构建一些东西并面临添加到应用程序 AngularJS 的问题。 Angular2 和 ReactJs 工作完美,但如果添加 AngularJS 并尝试为此应用程序使用
我是一名优秀的程序员,十分优秀!