前言:这篇博客我们重点讲 线性表中的顺序表、链表
线性表(linear list)是n个具有相同特性的数据元素的****有限序列。线性表是一种在实际中广泛使用的数据结构,**常见的线性表:**顺序表、链表、栈、队列...
线性表在逻辑上是线性结构 ,也就说是连续的一条直线 。但是在物理结构上并不一定是连续的 ,线性表在物理上存储时,通常以数组和链式结构的形式存储。
顺序表(ArrayList)
什么是顺序表?
顺序表是用一段物理地址连续 的存储单元,依次 存储数据元素的线性结构 ,一般情况下采用数组存储。在数组上完成数据的增删查改。
也就是说,顺序表 底层结构就是 一个数组。是使用数组 来完成的一种结构。
那现在有一个数组,如果有一组数据 1,2,3,4,5,6,7 , 我们现在想把这组数据放到数组里去。
data:image/s3,"s3://crabby-images/217a7/217a7aee4e930b3b1551ac9b613c27bc3d9db265" alt=""
我们先放进1,2,3 , 问当前数组里有多少个有效数据? 很明显是 3 个。
data:image/s3,"s3://crabby-images/c009a/c009a9083a0cfba40f0fe97b066fcc4d6726409c" alt=""
那怎么利用程序判断他有 3 个有效数据呢?
我们需要先定义一个变量 usedSize , 放进一个数据 usedSize++,放一个就加一下。这不就能确定有几个有效数据了吗。
代码实现 (MyArrayList)
接下来我们利用代码 来实现顺序表的增删查改等方法:
--- 打印顺序表
就是遍历数组
data:image/s3,"s3://crabby-images/46f59/46f59a89866757beddf5205552d710791db07111" alt=""
--- 新增元素
1.新增元素,默认在数组最后新增
data:image/s3,"s3://crabby-images/29e63/29e63083fa702655c080d40649a71ddf96695394" alt=""
这么写没问题吗?如果满了怎么办?
所有我们要写一个方法来判断是否满了 :
data:image/s3,"s3://crabby-images/db5d3/db5d3d5df7dbd2de0ec2d48024b8cec8bfd8350e" alt=""
新增元素前我们要判断是否满了,满了的话要****扩容
data:image/s3,"s3://crabby-images/5f145/5f145474e5393eda22bc29d23793b7928b2f3d7d" alt=""
新增看看效果:
data:image/s3,"s3://crabby-images/81e00/81e0009b4b0562a7442e6c6db84c70fa0c7d40a0" alt=""
data:image/s3,"s3://crabby-images/50186/5018672b0d69fe76cb3e31cf5e78b14d18a2ec7c" alt=""
2.在指定位置新增元素
我们的逻辑应该是把指定位置之后的数据 都向后挪,把指定位置空出来,再在指定位置插入数据 。
那具体该如何挪数据?
我们要从最后的位置开始挪 ,不能从第一个开始挪,不然会把之后的数据盖掉。
data:image/s3,"s3://crabby-images/01155/01155730ef63b73086bc2e89ee642dfb972db663" alt=""
data:image/s3,"s3://crabby-images/387a3/387a30044eb6610109a4cc5ef9c00d226dd75771" alt=""
data:image/s3,"s3://crabby-images/35d64/35d649ac378dbd9f30195ce27fb68713539fe418" alt=""
那我们现在确定了挪的方法,接下来我们看看代码如何实现:
data:image/s3,"s3://crabby-images/61949/6194975c3901de493c1e7563f5408e465a8ba80d" alt=""
注意:我们这里是要判断pos的位置是否合法的 ,
pos不能小于0,也不能跳着放(大于usedSize).
data:image/s3,"s3://crabby-images/69611/69611d0f8cdd65f31e9b54022c2db74bd7eeb9c6" alt=""
为此我们还搞了个异常
data:image/s3,"s3://crabby-images/e9992/e999277ffe37d0182c13fc1d7338a527940debc0" alt=""
(ps:异常之前讲过 链接 https://blog.csdn.net/iiiiiihuang/article/details/130671801?spm=1001.2014.3001.5501 )
我们看看运行效果 :
data:image/s3,"s3://crabby-images/9e792/9e792bdcb0f4b95ba16be26b546ae28b5236899b" alt=""
data:image/s3,"s3://crabby-images/53803/53803639fbe174afa57a661947f958129acf22dd" alt=""
看看位置不合法时的运行结果:
data:image/s3,"s3://crabby-images/c6443/c644326d776f231d4ca2fff74553d2a09b733d58" alt=""
data:image/s3,"s3://crabby-images/bf881/bf881ebaedec684da9233499f9f97a904a32c8e7" alt=""
--- 判断是否包含某个元素
data:image/s3,"s3://crabby-images/c5008/c5008129d7559fc118b3e454b0cb530f67059164" alt=""
运行结果
data:image/s3,"s3://crabby-images/6cae9/6cae9a117cd8a08da8d8381b77b78f60ef581f46" alt=""
data:image/s3,"s3://crabby-images/951f5/951f569090512b96570811d77cd80bb1b441cffd" alt=""
--- 查找某个元素具体位置(下标)
data:image/s3,"s3://crabby-images/a572f/a572ff58d5a3798a13a9935841dad99b324bef78" alt=""
运行结果
data:image/s3,"s3://crabby-images/89986/8998658bf8b80d5159d51ee861e64d95d0fe66c4" alt=""
data:image/s3,"s3://crabby-images/f5a93/f5a930e21cac700344ce20af6fefaa52d21297a6" alt=""
--- 获取 pos 位置的元素
data:image/s3,"s3://crabby-images/18eb6/18eb6ff2833263e34fb9ab1a98ec27f490260c68" alt=""
还要判断位置是否合法。
运行结果
data:image/s3,"s3://crabby-images/2d9d7/2d9d7f67974f697098e43389bff818b660fdb103" alt=""
data:image/s3,"s3://crabby-images/8c4e1/8c4e1bf75bf337c55cf59d6280e0fb4f30edd50f" alt=""
--- 给pos位置 的值设为 value
这里同样要先判断 pos 位置是否合法,那我们可以单独写一个方法,来判断。(方法的封装)
data:image/s3,"s3://crabby-images/b4156/b415603ac9c66cab85e009bd3644d3e8e823bb6a" alt=""
data:image/s3,"s3://crabby-images/05b88/05b88578c5e01aba775b6fd639e1700112c2ec34" alt=""
运行结果
data:image/s3,"s3://crabby-images/028bf/028bfaafd4e89fbb6b3f690435d080c7166f14bc" alt=""
data:image/s3,"s3://crabby-images/7a6f8/7a6f870d521ba4dbdd044d42f889272802f0339f" alt=""
--- 获取顺序表长度
data:image/s3,"s3://crabby-images/bba7c/bba7c1d5856e5ab33659c1cb49f154c75a842df2" alt=""
--- 删除第一次出现的关键字
代码实现
data:image/s3,"s3://crabby-images/9a589/9a589ee2e8c6d9df158db545537e302b4752fd58" alt=""
注意:usedSize - 1 防止越界, 这里是 this.elem[i] = this.elem[i + 1], 那 i 走到倒数第二位就行了
运行结果
data:image/s3,"s3://crabby-images/51c87/51c8732b95db79a7fe8329c1d1fcf5e404030a1b" alt=""
--- 清除顺序表
data:image/s3,"s3://crabby-images/2c826/2c82641e7390a1f446f6b6ff9b6ed3bb6d352266" alt=""
ps (小提一嘴): 现在的都是整型这么写可以,但是的是 引用类型 的时候要一个一个 置为 null ,删除也要有类似操作。
完整代码
java
import java.util.ArrayList;
import java.util.Arrays;
/**
* @Author: iiiiiihuang
*/
public class MyArrayList {
private int[] elem;//存放数据元素。
private int usedSize;//代表当前顺序表中有效数据个数。
private static final int DEFAULT_SIZE = 10;
/**
* 默认构造方法
*/
public MyArrayList() {
this.elem = new int[DEFAULT_SIZE];
}
/**
* 指定容量
* @param initCapacity
*/
public MyArrayList(int initCapacity) {
this.elem = new int[initCapacity];
}
/**
* 打印顺序表中的所有元素
*/
public void display() {
for (int i = 0; i < this.usedSize; i++) {
System.out.print(this.elem[i] + " ");
}
}
public boolean isFull() {
if(this.usedSize == this.elem.length){
return true;
}
return false;
}
/**
* 新增元素,默认在数组最后新增
* @param date
*/
public void add(int date){
if(isFull()){
//扩容
this.elem = Arrays.copyOf(this.elem, 2*this.elem.length);
}
this.elem[this.usedSize] = date;
this.usedSize++;
}
//判断pos位置是否合法
private void checkPos(int pos){
if(pos < 0 || pos >= usedSize){
throw new PosOutOfBoundsException(pos + "位置不合法");
}
}
/**
* 在指定位置(pos)新增元素
* @param pos
* @param date
*/
public void add(int pos, int date) {
//判断pos位置是否合法
if(pos < 0 || pos > usedSize){
throw new PosOutOfBoundsException(pos + "位置不合法");
}
if(isFull()){
this.elem = Arrays.copyOf(this.elem, 2*this.elem.length);
}
for (int i = this.usedSize - 1; i >= pos; i--) {
this.elem[i + 1] = this.elem[i];
}
this.elem[pos] = date;
usedSize++;
}
/**
* 判断是否包含某个元素
* @param toFind
* @return
*/
public boolean contains(int toFind){
for (int i = 0; i < this.usedSize; i++) {
if(this.elem[i] == toFind){
return true;
}
}
return false;
}
/**
* 查找某个元素具体位置(下标)
* @param toFind
* @return
*/
public int indexOf(int toFind){
for (int i = 0; i < this.usedSize; i++) {
if(this.elem[i] == toFind){
return i;
}
}
return -1;
}
/**
* 获取 pos 位置的元素
* @param pos
* @return
*/
public int get(int pos) {
//判断pos位置是否合法
checkPos(pos);
return this.elem[pos];
}
/**
* 给pos位置 的值设为 value
* @param pos
* @param value
*/
public void set(int pos, int value) {
checkPos(pos);
this.elem[pos] = value;
}
/**
* 获取顺序表长度
*/
public int size() {
return this.usedSize;
}
/**
* 删除第一次出现的关键字
* @param toRemove
*/
public void remove(int toRemove) {
//获取要删除元素下标
int index = indexOf(toRemove);
if(index == -1){
System.out.println("没有这个数据");
return;
}
//usedSize - 1 防止越界
for (int i = index; i < this.usedSize - 1; i++) {
this.elem[i] = this.elem[i + 1];
}
usedSize--;
}
/**
* 清除顺序表
*/
public void clear() {
this.usedSize = 0;
}
}
上述是自己实现一个顺序表结构,那以后用到顺序表都要我们自己重新实现吗? 当然不用啦!
Java里面已经帮你处理好了,有现成的 ,就是ArrayList
ArrayList
在集合框架中,ArrayList是一个普通的类,实现了List接口 。
data:image/s3,"s3://crabby-images/8bcd9/8bcd980c43427b0185b0dc8fae9b34898ade4b4f" alt=""
- ArrayList是以泛型方式实现的,使用时必须要先实例化
- ArrayList实现了RandomAccess接口,表明ArrayList支持随机访问
- ArrayList实现了Cloneable接口,表明ArrayList是可以clone的
- ArrayList实现了Serializable接口,表明ArrayList是支持序列化
- 和Vector不同,ArrayList不是线程安全的,在单线程下可以使用,在多线程中可以选择 Vector或者CopyOnWriteArrayList
- ArrayList底层是一段连续的空间,并且可以动态扩容,是一个动态类型的顺序表
我们可以在 IDEA 里看到 ArrayList 的源码
data:image/s3,"s3://crabby-images/c690f/c690f7ba4de19eaae43ef6a7b9dc80ca612a4389" alt=""
接下来我们 来介绍一下 ArrayList 的几种用法。
ArrayList 的实例化
data:image/s3,"s3://crabby-images/259a6/259a69d85a9b10c9e2dfd2bbea90a6dc596c200b" alt=""
这两种方法都行,区别就在于,方法一 可以调用的方法更多一些。 但是方法二 发生了向上转型,
一般情况下,我们用方法二多一点。
(ps: 向上转型 前面介绍过了 链接 https://blog.csdn.net/iiiiiihuang/article/details/130484383?spm=1001.2014.3001.5501 )
ArrayList的构造
|----------------------------------------|------------------------------|
| 方法 | 解释 |
| ArrayList() | 无参构造 |
| ArrayList(Collection<? extends E> c) | 利用其他 Collection 构建 ArrayList |
| ArrayList(int initialCapacity) | 指定顺序表初始容量 |
当我们调用不带参数的构造方法时,默认在第一次 add 时才会分配大小为10的内存
扩容按1.5倍进行扩容
ArrayList常见操作
|-----------------------------------------------|---------------------------|
| 方法 | 解释 |
| boolean add(E e) | 尾插 e |
| void add(int index, E element) | 将 e 插入到 index 位置 |
| boolean addAll(Collection<? extends E> c) | 尾插 c 中的元素 |
| E remove(int index) | 删除 index 位置元素 |
| boolean remove(Object o) | 删除遇到的第一个 o |
| E get(int index) | 获取下标 index 位置元素 |
| E set(int index, E element) | 将下标 index 位置元素设置为 element |
| void clear() | 清空 |
| boolean contains(Object o) | 判断 o 是否在线性表中 |
| int indexOf(Object o) | 返回第一个 o 所在下标 |
| int lastIndexOf(Object o) | 返回最后一个 o 的下标 |
| List<E> subList(int fromIndex, int toIndex) | 截取部分 list |
注意:这个remove怎么用?
data:image/s3,"s3://crabby-images/5e2d9/5e2d9606382efea2ae275127c209404073d2add6" alt=""
要这么写才能删掉 等于2 的元素
data:image/s3,"s3://crabby-images/079c2/079c2c405a106e0caa3f41a08162a1d9fcdc2df5" alt=""
data:image/s3,"s3://crabby-images/82b07/82b0716583c89db6b9c4b8efcead91b42c812b1a" alt=""
演示一下 subList
data:image/s3,"s3://crabby-images/6c944/6c944379b2e2a23754ef29b9164fefdaeeeb8651" alt=""
data:image/s3,"s3://crabby-images/16f59/16f59788de41d525c6a963322b2d93e2fe32c787" alt=""
截取到2,3 (Java里一般都是左闭右开的 [ , , ) )
data:image/s3,"s3://crabby-images/43f23/43f23fcaa6f437dc08932efdd99ab78e1c344c27" alt=""
如果我们把 list3 的 0 下标该成 188 会方生什么?
data:image/s3,"s3://crabby-images/de1dd/de1ddf4c30c0cb7f1e25f88ee6977eb0ab9822bd" alt=""
data:image/s3,"s3://crabby-images/00c64/00c64a2dd0ccba2a10d5e1b84555d7926d690c7b" alt=""
我们发现,list2 的 1 下标的位置(list3 的 0下标)也变成了188 . 为什么?
这是因为list3, 截取的时候并没有真正的把这个数据拿出来,只是指向了1 下标那儿的地址 ,所有更新值肯定会影响 list2.
data:image/s3,"s3://crabby-images/b227a/b227a0ff306cf455311995dbdbedfd874b10d977" alt=""
ArrayList的遍历
ArrayList 可以使用三方方式遍历:使用迭代器 ,for循环+下标、foreach
上面是 直接 用sout 遍历 的,是因为重写了 toString 方法**。**
data:image/s3,"s3://crabby-images/ac545/ac545d366625fc4bae5a4f794dc9c9ca1dc6b066" alt=""
---- 迭代器
一般情况下,能够直接通过 sout 输出 引用指向对象当中的内容的时候,此时一定重写了 toString 方法****。
data:image/s3,"s3://crabby-images/52a2f/52a2f847e240a7c084c2dd03dc031497f162568b" alt=""
那我们看看ArrayList 里有没有 toString (ctrl + F) 搜索一下。发现没有。
data:image/s3,"s3://crabby-images/78cf7/78cf7924857aa1384c157b4daae1ed6f3a9efcf2" alt=""
但是 ArrayList 还继承了 AbstractList,我们去 AbstractList 里找找。发现还没有。
data:image/s3,"s3://crabby-images/03203/0320345f9307c5b209609768491c33bff264aebf" alt=""
但是 AbstractList 还继承了 AbstractCollection ,
我们在 AbstractCollection 这里找到了 toString。
data:image/s3,"s3://crabby-images/9dffb/9dffbe2d4d8a0466f308f8ac90e3f6b24d39a986" alt=""
下面画线那个部分就是迭代器(遍历当前集合)
data:image/s3,"s3://crabby-images/4de05/4de053a9035075ca4282a5dad9110432bf01ac4e" alt=""
--- 用法一
data:image/s3,"s3://crabby-images/d15dd/d15ddda19e51c95ce61c1675164f28dcc42697a1" alt=""
--- 用法二
data:image/s3,"s3://crabby-images/1f40f/1f40f05bebc6e17063c88bd3d1929fbfba8e7103" alt=""
上面两个用那个都行
--- 从后往前打印
data:image/s3,"s3://crabby-images/6be74/6be749317a11d13ee65961dadbcedb47ea64da56" alt=""
data:image/s3,"s3://crabby-images/8a44d/8a44d6f63bcb5d83f6bbc9ba724127caf5ce9bd4" alt=""
data:image/s3,"s3://crabby-images/f9be1/f9be18c6609eb9fb35a5f89f13119daa491ae471" alt=""
ps : 这个方法不行,因为这个不能传参
data:image/s3,"s3://crabby-images/a7a14/a7a14057eb5d9ed18f981743c52477f600a791de" alt=""
data:image/s3,"s3://crabby-images/d7686/d76861d35617623bcba9de8bb2136acf3025b00a" alt=""
---- for循环+下标
data:image/s3,"s3://crabby-images/f7918/f7918a2cada93cd62af55b61e54e5a9210bb561b" alt=""
data:image/s3,"s3://crabby-images/2e360/2e3603b0f4fd2308053fb2b5770b3bc833803391" alt=""
---- foreach
data:image/s3,"s3://crabby-images/f5ce2/f5ce262128839f24ee3ea40c44c81305e25bfaf3" alt=""
data:image/s3,"s3://crabby-images/13934/13934cc0d1c6af5b62730182a4122269425da1c6" alt=""
ArrayList 的优缺点
优点
1.可以通过下标 进行随机访问,顺序表适合对静态的数据进行 查找 和 更新
缺点
1.添加元素的效率比较低 (假如在 0 位置添加元素,就需要把后面所有的元素都往后移)
2.删除的效率也低(假如删除 0 位置元素,也需要移动后面所有的元素)
3.扩容 按1.5 倍扩容,有浪费 空间的情况。
(顺序表不适合用来 插入和删除 数据)
**顺序表适合对静态的数据进行 查找 和 更新,不适合用来 插入和删除 数据,这时候就需要用链表来做。**接下来我们来介绍链表。
链表 (LinkedList)
什么是链表?
链表 是一种 物理存储结构上非连续 的存储结构,数据元素的 逻辑顺序 是通过链表中的 引用链接次序实现的 。
链表是由一个一个 节点 连成的 :(这个是单向 不带头 非循环 链表的节点)
data:image/s3,"s3://crabby-images/47e76/47e76e006efeeead5ff6f724cb0c2a4a62620d74" alt=""
单向 不带头 非循环 链表
data:image/s3,"s3://crabby-images/112cc/112ccd71878a7b82398637d8ff6c973d68b17f18" alt=""
除此之外还有很多 类型的 链表,一般从三个方向去分
- 单向 双向
- 不带头 带头
- 非循环 循环
经过排列组合就会有下面这几种类型的链表:
- 单向 不带头 非循环 链表
- 单向 不带头 循环 链表
- 单向 带头 非循环 链表
- 单向 带头 循环 链表
- 双向 不带头 非循环 链表
- 双向 不带头 循环 链表
- 双向 带头 非循环 链表
- 双向 带头 循环 链表
上面我们画了不带头的 链表,接下来我们了解一下 带头的是啥样的?
data:image/s3,"s3://crabby-images/f7ec7/f7ec7984b33d1f669a24acc047dc63234830362f" alt=""
再来看看 循环的 :
data:image/s3,"s3://crabby-images/24412/24412bb0bf49e4ea23f9f03ba7ec907a04654a22" alt=""
等我们把单向的介绍完,在介绍双向的 。
我们重点讲单向 不带头 非循环 链表和双向 不带头 非循环 链表(这两种是工作,笔试面试考试重点)
单向 不带头 非循环 链表
代码实现
节点
上述可知 链表是由一个一个 节点 组成的,所以我们可以把这个节点定义成个内部类 。
data:image/s3,"s3://crabby-images/aba3f/aba3fcbb0cdddf0fc81829eb232d1c089b6e3130" alt=""
还得有一个 指向 第一个节点 的属性:
data:image/s3,"s3://crabby-images/64836/6483667fce644fdf58079dc50f801ed16e48105b" alt=""
插入数据
--- 头插法
data:image/s3,"s3://crabby-images/0a2d6/0a2d657ef0faebee88f0334f52feef8111eddcf7" alt=""
data:image/s3,"s3://crabby-images/76a31/76a31dc16b9b62cd1a31645e988e91abfe034f6f" alt=""
注意:如果我们把下面的语句反过来写可以吗? 变成 head = node; node.next = head;
data:image/s3,"s3://crabby-images/6ced7/6ced7216eff4cadff2cd85edaaf1100360d9548b" alt=""
不可以,先把node 置为 head, node 的下一个指向 head, 那不就是 自己指向自己了吗,后面的节点就都丢失了。
我们插入看看效果
data:image/s3,"s3://crabby-images/db9c9/db9c96e24842d38c0890f3f8734456c5e51c5b10" alt=""
data:image/s3,"s3://crabby-images/fa1e7/fa1e7c131ed30aefc9ec0d8360b5212845dd3883" alt=""
--- 尾插法
先找到最后一个节点,
data:image/s3,"s3://crabby-images/05603/05603264e3387b06363f6d0fc50a9ead52abf64b" alt=""
所以循环里要写成 cur.next != null , 在到最后一个节点这就不进入循环了,那此时cur 就是最后一个节点,而不能写成cur != null, 这样会遍历完整个链表,停不住。
data:image/s3,"s3://crabby-images/5f972/5f972e0698a8a1ba4361ccbdc4bab5777761af40" alt=""
运行看看效果:
data:image/s3,"s3://crabby-images/a6f53/a6f53f3b9e5a450ab05b5c92df69cc2008399d8e" alt=""
data:image/s3,"s3://crabby-images/29d45/29d45fcc05890a1646354c1a1fd153fcddb1e4ff" alt=""
似乎是可以的,但是如果现在链表了一个节点也没有,再插入呢?
data:image/s3,"s3://crabby-images/6735c/6735c2355e13df0170dec0c0f45d190247598af8" alt=""
**报错了!!------ 空指针异常 ,**为啥呀? 我们返回代码哪里看一看,调试一下:
data:image/s3,"s3://crabby-images/c83b1/c83b1da30c94b50788dd4f268d5b2e667a5fc9c9" alt=""
data:image/s3,"s3://crabby-images/6c8f7/6c8f722f2e790901fa5d42f4031e91a953a83796" alt=""
我们发现当 链表里一个节点都没有的时候,head = null,那cur = head, cur 也是 null,
那cur都为空了,哪来的 next ,那当然就报出来空指针异常了。
所以接下来我们改进一下代码:
data:image/s3,"s3://crabby-images/eea97/eea979eb074c7f67b5d1ccfd359d78966dafc5da" alt=""
那我们再运行一下代码,就没有任何问题了
data:image/s3,"s3://crabby-images/fa33c/fa33c06bd3311aca14b1b40625653b3c5c9d127c" alt=""
--- 任意位置插入(第一个数据节点为0号下标)
data:image/s3,"s3://crabby-images/f9880/f9880da0e2dfd7816e68bd11b4349c390554fb9e" alt=""
很明显,我们要确定 插入位置的前一个节点 , 那如果你要插入 2 位置,那你就要找到 1 位置节点,那从 0 位置到 1 位置要走一步,也就是 index - 1 步。
**找到要插入位置的前一个节点。**这个是写方法 里面,还是单独封装看你喜好。
(我用的 while 循环,如果用for 循环的话,i < index - 1, 我觉得while 循环好理解点。)
data:image/s3,"s3://crabby-images/dea21/dea21d7381380f22daf2f4e14494c42ee2c74cc4" alt=""
找到位置了,我们就可以插入了。
先判断插入位置合不合法(类比上面的顺序表的部分)
data:image/s3,"s3://crabby-images/7aca6/7aca651e8b000a901b2fc720896a898654516280" alt=""
都要先和后边的节点建立联系哦
看看效果
data:image/s3,"s3://crabby-images/1cc19/1cc19f535ecb38e8cc452e47424165ddb2be43da" alt=""
data:image/s3,"s3://crabby-images/a1475/a1475d8f19f7ff09a5640aa480c9ec3d46419088" alt=""
打印链表
data:image/s3,"s3://crabby-images/e2dbe/e2dbe5b8f8e210c0f411f8ba64be1269f50e39bc" alt=""
但是用上面的方式走完之后,我自己都不知道头节点在哪里了 ,很可怕耶,
所以我们定义一个 cur节点 来代替 头节点 往下走。
data:image/s3,"s3://crabby-images/3d9dd/3d9ddc946f6cd8b46cd5bae0e6ac88c5313e0c4d" alt=""
data:image/s3,"s3://crabby-images/b3631/b3631592ee518fbbbf845c52fbbb63987bb200cb" alt=""
单链表的长度
data:image/s3,"s3://crabby-images/bea7d/bea7db84969a453666de60dcfe6a001d9e62e503" alt=""
查找单链表中是否包含关键字key
data:image/s3,"s3://crabby-images/d9db4/d9db4304697dd0b6665017a30ccf3e6e5a7734a5" alt=""
删除第一次出现关键字为key的节点
data:image/s3,"s3://crabby-images/ac8bc/ac8bc7af5adf3e5138fb6de3c27c18f7058caf53" alt=""
还是要找到删除节点的前一个,
为什么,循环那里要 cur.next != null, 因为,cur.next 不能是null,因为如果它是 null 的话,
那cur.next.val 就找不到值,那就会引起空指针异常,所以只要走到倒数第一个停住就好了(cur走到最后一个节点时不进入循环)
data:image/s3,"s3://crabby-images/2b008/2b00894f7b10cc87dfe7c9abfb8d80d3b9c25d50" alt=""
data:image/s3,"s3://crabby-images/f51d9/f51d92fc26d13867a4d1f56593b039844d46c4d2" alt=""
看看效果:
data:image/s3,"s3://crabby-images/56c20/56c20a73924a6db75e3166ac6ef5b39292084fb5" alt=""
删除所有值为key的节点
data:image/s3,"s3://crabby-images/72532/72532b1d8ce0072b30c88af11c31fb92ba8f5b8b" alt=""
注意 :我们删除的是 cur 对应的值,cur 从第二个节点开始走,那如果第一个也是23(要删除的值)呢 ,所以我们要单独删除头节点,且在最后,不能放在前面。
data:image/s3,"s3://crabby-images/b203d/b203d9639b65302ca4ce7c449bbbb08d308f98f9" alt=""
data:image/s3,"s3://crabby-images/c6bb2/c6bb280e75fbb287baf927337843d96fe5fcc6d1" alt=""
运行结果
data:image/s3,"s3://crabby-images/832cb/832cb1b7c4436c4ff87603ea59889d37deee53d6" alt=""
清除链表
data:image/s3,"s3://crabby-images/d6262/d6262eabed7aaa8a291356403f0b3b0ec0bd47a2" alt=""
完整代码
java
/**
* @Author: iiiiiihuang
*/
public class MySingleLinkedList {
//把节点定义成个内部类
static class ListNode {
public int val;//节点的值域
public ListNode next;//下一个节点的地址
public ListNode(int val) {
this.val = val;
}
}
public ListNode head;//头节点
/**
* 头插法
* @param data
*/
public void addFirst(int data) {
//给插入的数据创个节点
ListNode node = new ListNode(data);
node.next = head;
head = node;
}
/**
* 尾插法
* @param data
*/
public void addLast(int data) {
ListNode node = new ListNode(data);
ListNode cur = head;
if(cur == null) {
head = node;
return;
}
//先找到最后一个节点
while (cur.next != null) {
cur = cur.next;
}
cur.next = node;
}
/**
* 打印链表
*/
public void display() {
ListNode cur = head;
while(cur != null) {
System.out.print(cur.val + " ");
cur = cur.next;
}
System.out.println();
}
/**
* 任意位置插入
* @param index
* @param data
*/
public void addIndex(int index,int data) {
ListNode node = new ListNode(data);
//先判断插入位置合不合法
if(index < 0 || index > size()){
throw new IndexOutOfBoundsException();
}
if(index == 0) {
addFirst(data);
return;
}
if(index == size()) {
addLast(data);
return;
}
ListNode cur = findIndexSubOne(index);
node.next = cur.next;
cur.next = node;
}
//找到要插入位置的前一个节点
private ListNode findIndexSubOne(int index) {
ListNode cur = head;
while (index - 1 != 0) {
cur = cur.next;
index--;
}
return cur;
}
/**
* 单链表的长度
* @return
*/
public int size() {
ListNode cur = head;
int count = 0;
while (cur != null) {
count++;
cur = cur.next;
}
return count;
}
/**
* 查找单链表中是否包含关键字key
* @param key
* @return
*/
public boolean contains(int key) {
ListNode cur = head;
while (cur != null) {
if(cur.val == key) {
return true;
}
cur = cur.next;
}
return false;
}
/**
* 删除第一次出现关键字为key的节点
* @param key
*/
public void remove(int key) {
if(head == null) {
return;
}
//单独删除头节点
if(head.val == key) {
head = head.next;
return;
}
//找到删除节点的前一个
ListNode cur = findRemSubOne(key);
if(cur == null) {
System.out.println("没有你要删除的数字");
return;
}
ListNode rem = cur.next;
cur.next = rem.next;
}
//找到删除节点的前一个节点
private ListNode findRemSubOne(int key) {
ListNode cur = head;
//这里跳过了头节点
while (cur.next != null) {
if (cur.next.val == key) {
return cur;
}
cur = cur.next;
}
return null;
}
/**
* 删除所有值为key的节点
* @param key
*/
public void removeAllKey(int key) {
ListNode cur = head.next;
ListNode prev = head;
while (cur != null) {
if(cur.val == key) {
prev.next = cur.next;
} else {
prev = cur;
}
cur = cur.next;
}
//删除头节点
if(head.val == key) {
head = head.next;
}
}
/**
* 清除链表
*/
public void clear() {
this.head = null;
}
}
双向 不带头 非循环 链表
画图看看结构
data:image/s3,"s3://crabby-images/c54bc/c54bcefe475f69f2fdf444190f7d7543db3b97b7" alt=""
代码实现
创建节点内部类
data:image/s3,"s3://crabby-images/bd666/bd666626d341fdf037231d747743c0bce96a7145" alt=""
data:image/s3,"s3://crabby-images/a5ff4/a5ff42c5ca360f22439e2b758bb8e729e06cc7f3" alt=""
上面有的方法这也有 😀😀
插入数据
--- 头插法
data:image/s3,"s3://crabby-images/dee5e/dee5ebcd9d4b52f600241f6bd6ef2d25e6aabc4d" alt=""
链表为空时,必须单独分情况,因为如果不分会:
data:image/s3,"s3://crabby-images/b145c/b145c8e7606ead2e0d8c960828b5a76d7bb21c0a" alt=""
运行结果
data:image/s3,"s3://crabby-images/e3e33/e3e3327027d28c74a94f07f3569db5b1db864465" alt=""
data:image/s3,"s3://crabby-images/27cd9/27cd931e3fa20bde94ecd828385be046c659fa47" alt=""
--- 尾插法
data:image/s3,"s3://crabby-images/21a72/21a721da6642187b0dea4918ea135051712725cb" alt=""
data:image/s3,"s3://crabby-images/c06f8/c06f85275330f48da7b24bba7897cb857699e996" alt=""
打印链表
和上面一样
data:image/s3,"s3://crabby-images/3fb88/3fb8894009b7efc84de7f81f6e5b6f3b888b8fb5" alt=""
查找链表中是否包含关键字key
和上面一样
data:image/s3,"s3://crabby-images/33c17/33c17ac544e5c804382a9aa452640aafebdfe910" alt=""
链表的长度
和上面一样
data:image/s3,"s3://crabby-images/84917/8491789f0da7a187477bc3768f66874385c9d417" alt=""
运行结果
data:image/s3,"s3://crabby-images/f3fab/f3fabd14601415d45615e7ab10539604138671a7" alt=""
data:image/s3,"s3://crabby-images/9708f/9708f5e69b5d628087f619ed25c87749f34bd9c2" alt=""
任意位置插入,第一个数据节点为0号下标
data:image/s3,"s3://crabby-images/5e05f/5e05f90436909bfbd32f4c8b44f7805acf690ef1" alt=""
data:image/s3,"s3://crabby-images/a4857/a485733bc500455ef5de11a4fad72c2fb048b5a7" alt=""
运行结果
data:image/s3,"s3://crabby-images/2e720/2e7209248cde46e2abaab78b42b38be8e210740f" alt=""
删除第一次出现关键字为key的节点
data:image/s3,"s3://crabby-images/39582/395827a88131aa98a5c1ea06d113bfa1826776ff" alt=""
代码
data:image/s3,"s3://crabby-images/d6c39/d6c39a88d65c8dd36362a30fb016c10c96b17b9e" alt=""
头尾分开来,不然会空指针异常 。
运行结果
data:image/s3,"s3://crabby-images/b2c22/b2c2285f14e24507e1c0f6b38064cef2ba0a0a07" alt=""
data:image/s3,"s3://crabby-images/d5c4d/d5c4df314ffba7b8a7e83d44ed8f3d787e756fbe" alt=""
data:image/s3,"s3://crabby-images/bc630/bc630acb53e397077b88a89fbf89461ca79d34b6" alt=""
但是上面的代码还有问题
data:image/s3,"s3://crabby-images/42746/42746c14b1ae9b0af51156614f68d428323afc5d" alt=""
如果只有一个节点时,head = head.next; 那此时head 就是 null,那此时再head.prev 就会引起空指针异常
所以要改成这样:
data:image/s3,"s3://crabby-images/4a8bf/4a8bfe583defb65a220214d0368d271893c23ddf" alt=""
data:image/s3,"s3://crabby-images/b78c1/b78c10b0f19fddad51170feff7158a394b0c13c6" alt=""
删除所有值为key的节点
和上面几乎一致,只有一点不一样。
data:image/s3,"s3://crabby-images/cc738/cc738c460a5ea303b35e83dd62174e450368ba7b" alt=""
运行结果
data:image/s3,"s3://crabby-images/a194a/a194ab538b549ec9a2d0368553fadfff3fa391ac" alt=""
清除链表
最简单粗暴的
data:image/s3,"s3://crabby-images/f6075/f6075c1758176a3d94037da443594bfd41c292e5" alt=""
data:image/s3,"s3://crabby-images/0f217/0f21727b005f3f4563423e29d2a6e5ce603fd01e" alt=""
或者把每一个都置为空
data:image/s3,"s3://crabby-images/0807e/0807e807f74e89635c7172c8e21f6eba9601fae6" alt=""
完整代码
java
import java.util.List;
/**
* @Author: iiiiiihuang
*/
public class MyLinkedList {
static class ListNode {
private int val;
private ListNode prev;
private ListNode next;
public ListNode(int val) {
this.val = val;
}
}
public ListNode head;
public ListNode last;
/**
* 头插法
* @param data
*/
public void addFirst(int data){
ListNode node = new ListNode(data);
if(head == null) {
head = node;
last = node;
}else {
node.next = head;
head.prev = node;
head = node;
}
}
/**
* 尾插法
* @param data
*/
public void addLast(int data){
ListNode node = new ListNode(data);
if(head == null) {
head = node;
last = node;
} else {
last.next = node;
node.prev = last;
node = last;
}
}
/**
* 任意位置插入,第一个数据节点为0号下标
* @param index
* @param data
*/
public void addIndex(int index,int data){
ListNode node = new ListNode(data);
checkIndex(index);
if(index == 0) {
addFirst(data);
return;
}
if(index == size()) {
addLast(data);
return;
}
ListNode cur = searchIndex(index);
node.prev = cur.prev;
node.next = cur;
cur.prev.next = node;
cur.prev = node;
}
//找到插入位置
private ListNode searchIndex(int index) {
ListNode cur = head;
while (index != 0) {
cur = cur.next;
index--;
}
return cur;
}
//检查index的位置是否合法
private void checkIndex(int index) {
if(index < 0 || index > size()) {
throw new IndexOutOfBoundsException("位置不合法");
}
}
/**
* 查找关键字key是否在链表当中
* @param key
* @return
*/
public boolean contains(int key){
ListNode cur = head;
while (cur != null) {
if(cur.val == key) {
return true;
}
cur = cur.next;
}
return false;
}
/**
* 删除第一次出现关键字为key的节点
* @param key
*/
public void remove(int key){
ListNode cur = head;
while (cur != null) {
if(cur.val == key) {
//删除头节点
if(cur == head) {
head = head.next;
if(head != null) {
//只有一个节点时
head.prev = null;
} else {
last = null;
}
} else {
//删除中间节点 和 尾巴节点
if(cur.next != null) {
//删除中间节点
cur.prev.next = cur.next;
cur.next.prev = cur.prev;
} else {
cur.prev.next = cur.next;
last = last.prev;
}
}
return;
} else {
cur = cur.next;
}
}
}
/**
* 删除所有值为key的节点
* @param key
*/
public void removeAllKey(int key){
ListNode cur = head;
while (cur != null) {
if(cur.val == key) {
//删除头节点
if(cur == head) {
head = head.next;
if(head != null) {
//只有一个节点时
head.prev = null;
} else {
last = null;
}
} else {
//删除中间节点 和 尾巴节点
if(cur.next != null) {
//删除中间节点
cur.prev.next = cur.next;
cur.next.prev = cur.prev;
} else {
cur.prev.next = cur.next;
last = last.prev;
}
}
}
cur = cur.next;
}
}
/**
* 得到链表的长度
* @return
*/
public int size(){
ListNode cur = head;
int count = 0;
while (cur != null) {
count++;
cur = cur.next;
}
return count;
}
/**
* 打印链表
*/
public void display(){
ListNode cur = head;
while (cur != null) {
System.out.print(cur.val + " ");
cur = cur.next;
}
System.out.println();
}
/**
* 清除链表
*/
public void clear(){
ListNode cur = head;
while (cur != null) {
//先记录下数据
ListNode curNext = cur.next;
cur.prev = null;
cur.next = null;
cur = curNext;
}
head = null;
last = null;
}
}
LinkedList的使用
先实例化一下
data:image/s3,"s3://crabby-images/23ac8/23ac8449fa0fc432943a4b4f964563adb11134cb" alt=""
data:image/s3,"s3://crabby-images/f7c98/f7c98b4135305b2cb99011e91e16977f9c0c7d5d" alt=""
我们之前写的方法他都有,你只要调用就好了。😁😁😁
data:image/s3,"s3://crabby-images/d08bf/d08bf9577926ca6819634d908930231ccd724c44" alt=""
LinkedList 的 常见方法
|-----------------------------------------------|---------------------------|
| 方法 | 解释 |
| boolean add(E e) | 尾插 e |
| void add(int index, E element) | 将 e 插入到 index 位置 |
| boolean addAll(Collection<? extends E> c) | 尾插 c 中的元素 |
| E remove(int index) | 删除 index 位置元素 |
| boolean remove(Object o) | 删除遇到的第一个 o |
| E get(int index) | 获取下标 index 位置元素 |
| E set(int index, E element) | 将下标 index 位置元素设置为 element |
| void clear() | 清空 |
| boolean contains(Object o) | 判断 o 是否在线性表中 |
| int indexOf(Object o) | 返回第一个 o 所在下标 |
| int lastIndexOf(Object o) | 返回最后一个 o 的下标 |
| List<E> subList(int fromIndex, int toIndex) | 截取部分 list |
LinkedList 的总结
- LinkedList实现了List接口
- LinkedList的底层使用了双向链表
- LinkedList没有实现RandomAccess接口,因此LinkedList不支持随机访问
- LinkedList的任意位置插入和删除元素时效率比较高,时间复杂度为O(1)
- LinkedList比较适合任意位置插入或删除的场景
ArrayList和LinkedList的区别(顺序表和链表的区别)(面试题)
|-----------|--------------------|--------------------------|
| 不同点 | ArrayList | LinkedList |
| 存储空间上 | 物理上一定连续 | 逻辑上连续,但物理上不一定连续 |
| 随机访问 | 支持O(1) (有下标) | 不支持:O(N) |
| 头插 | 需要搬移元素,效率低O(N) | 只需修改引用的指向,时间复杂度为O(1) |
| 插入 | 空间不够时需要扩容 | 没有容量的概念 |
| 应用场景 | 元素高效存储+频繁访问时 | 任意位置插入和删除频繁时 |
╰(*°▽°*)╯╰(*°▽°*)╯╰(*°▽°*)╯╰(*°▽°*)╯╰(*°▽°*)╯完 ╰(*°▽°*)╯╰(*°▽°*)╯╰(*°▽°*)╯╰(*°▽°*)╯╰(*°▽°*)╯