LinKedList与链表
?
1. 链表
?
1.1 链表的概念与结构
? 链表 : 就像火车一样, 是节点(车厢)构成的 , 每个节点之间会连接起来。
? 图示:
data:image/s3,"s3://crabby-images/d5226/d52265f3f1b5d6a63f6a271a168b5f0308754052" alt="查看源图像"
? 注意 : 我们的链表 虽然 看起来是 节点上全部连接起来的如上面的火车一样一节一节的串起来,但是在物理的成面上, 链表其实是不连续的。
? 图解 :
data:image/s3,"s3://crabby-images/89795/8979500b3778b5668ab6c79739d6c5c04d103d8b" alt="在这里插入图片描述"
? 关于链表 , 我们现在知道了他是,通过 next 域,来进行 连接的, 那么接下来我们来 了解一下我们 的 链表结构 , 总共由 八种 ,但是不要担心 我们主要学两种即可 。
?
1.2 链表的结构
关于链表 : 我们由 这几个 关键字 单向 , 双向 和 带头 , 不带头 还有 循环 , 不循环
将这几个关键字 排列组合 ,能得到以下
-
单向带头不循环 -
单向带头循环 -
双向带头不循环 -
双向带头循环 -
单向不带头不循环 -
单向不带头循环 -
双向不带头不循环 -
双向不带头循环
? 这里就 看到了我们 链表的 8种结构 , 我们主要学习 单项不带头不循环 和 双向不带头不循环 因为后面我们面试中只要考察的是这两种。
? 2.1 单向带头不循环 和 单向不带头不循环 data:image/s3,"s3://crabby-images/df1b0/df1b03026f36687a28f6807922f9a262e3bf8111" alt="在这里插入图片描述"
2.2 单向带头循环 和 单向不带头循环 data:image/s3,"s3://crabby-images/49c41/49c4181029f59fe37ba571ba20358dd5cfbf761e" alt="在这里插入图片描述"
?
2.3 双向带头不循环 和 双向不带头不循环
data:image/s3,"s3://crabby-images/ef320/ef320616c7f2c8807452fbb28f1fcba8bf983f02" alt="在这里插入图片描述"
2.4 双向带头循环 和 双向不带头循环
? 这两个就跟 单向带头循环 和 单向不带头循环一样,只不过多了一个last 域而已
data:image/s3,"s3://crabby-images/09abe/09abe47d76c8d13cdd4d7d05adc30cb034c84cfe" alt="在这里插入图片描述"
? 链表的结构看完, 那么我们下面继续 , 这里还是通过实现 我们自己的链表 , 来慢慢了解我们的链表。
?
2. 链表实现
?
主要实现的方法
public class SingleLinkedList {
public void addFirst(int data);
public void addLast(int data);
public boolean addIndex(int index,int data);
public boolean contains(int key);
public void remove(int key);
public void removeAllKey(int key);
public int size();
public void display();
public void clear();
}
? 这里我们先来创建 一个链表, 这里采用的 比较low 的方法, 直接穷举 。 (先不使用插入的方法)
? 第一步创建一个 MySingleList 类 并且在这个类里面创建一个 内部类(静态也可以) ,这个类是专门的节点类用来创建我们的节点对象。
data:image/s3,"s3://crabby-images/b61fc/b61fc0597a25c3dc749a800e4e4b5875fac96ace" alt="在这里插入图片描述"
? 第二步 : 通过 create 创建我们的链表 ? 注意 : 我们还要创建一个 head 头节点 (用来存放第一个节点的地址)
data:image/s3,"s3://crabby-images/8edfa/8edfa0d2337ea13806c76d3f23e02a6432a070fc" alt="在这里插入图片描述"
? 第三步: 有了链表 接下来我们就来实现我们链表中的方法
?
2.1 打印链表
? 这个方法 可以说 非常简单, 就是创建一个 ListNode cur 让他引用 head 这个引用的 对象,然后 遍历每一个节点 然后打印 。
? 这里主要就只有两个问题 :
1、如何走到下一个节点
解 : cur = cur.next
2、什么时候把链表遍历完
解 : cur != null
?
图解 :
data:image/s3,"s3://crabby-images/a4100/a4100188c4995a1588afb6e4ceae6915924d87ba" alt="在这里插入图片描述"
? 注意 : 这我们需要严谨,如果我们的 链表为空我们还能 去打印吗 ? 这里就可以直接返回或抛出异常
data:image/s3,"s3://crabby-images/56cb8/56cb85de9af563545db0449d9ddf72618e60dacd" alt="在这里插入图片描述"
打印方法实现完 ,下面实现 求链表长度 。
?
2.2 返回链表长度
? 这个方法, 就只需要返回链表中节点的个数 , 这里同样是需要一个循环, 此时就需要思考 我们的循环结束的条件 。
?
想一想 : 我们是不是要数节点的个数,那么是不是需要走到每一个节点,那么 这里同样是要创建一个 cur 让他引用 head的对象,然后结束条件是不是就和打印一样cur != null 如果是 cur.next != null 那么 最后一个节点同样是没有记录的 。 ? 分析完我们就可以直接来写代码 :
data:image/s3,"s3://crabby-images/cb349/cb3490eb2266b27c3239d90468771beb430433b5" alt="在这里插入图片描述"
? 另外 :还有一种方法 ,看到这个size 了吗 ? 我们就可以在 后面些的新增方法中使用,每次添加一个元素 然后 让``size ++` 最后返回的size 就是我们的 链表长度
data:image/s3,"s3://crabby-images/c8212/c8212c6a2cf5288e1b2ca14f418c8ec0af72a842" alt="在这里插入图片描述"
? 这里这个方法就写完了, 下面来完成我们的 contains 方法 判断 key 是否存在单链表中
?
2.3 查找 key 元素是否存在链表中
? 这个方法我们同样需要使用循环, 这里同样是要遍历 整条链表,那么 我们循环结束的条件就是 cur != null ,同理 我们不能使用head 去遍历, 所以需要创建一个 cur 来代替 head , 这里我们 判断是否纯在,那么 拿 每个节点的value 与当前的 key 比较即可 相等放回 true , 不像等返回 false (注意这里我们是 int 类型,所以 可以直接使用 == , 如果是包装类 ,需要使用 equals 如果是 自定义类型 重写 equals 即可)
?
代码如下 :
data:image/s3,"s3://crabby-images/bfc04/bfc040f7af7cd83bd33a838565a97b7328f73746" alt="在这里插入图片描述"
?
这里我们是使用非常low 的方法创建了一条链表, 完成了上面这些方法, 其实你创建链表这几个方法 也能实现为啥我创建了呢? ? 是因为 我们可以写完一个方法 在main函数里面去测试 ,但是 上面我并没有去测试 , 这里可以去测试一下 。
? 下面就来学习一下新增方法, 头插和尾插 然后真正的创建一条链表.
?
2.4 新增方法 add
? 新增方法 总共有 3 个 一个是头插, 一个是尾插 ,一个是 任意位置插入
?
2.4.1 头插
? 图解 :
data:image/s3,"s3://crabby-images/800e3/800e39dc3d37fe77a5b67b378495042127d38588" alt="在这里插入图片描述"
? 代码实现 :
data:image/s3,"s3://crabby-images/ae249/ae2497d583d781517516c94636dba44525f4d56f" alt="在这里插入图片描述"
? 这里 如果 我们的 链表为空, 也不要紧 , 你想 head == null , 那么 cur.next = null , cur.next 本来就是null 在赋值一个 null 也是没有问题的, 最后在让 这个 cur 变为新的head
? 补充: 对比一下 顺序表的插入 ,链表的插入是不是就非常高效, 我们链表插入的数据是不是就不需要挪动 元素,只需要改变指向即可。
?
2.4.2 尾插
? 图解 : data:image/s3,"s3://crabby-images/b3ff1/b3ff17636e7b0aff13012d8e6e1390b074c50552" alt="在这里插入图片描述"
? 代码实现 :
data:image/s3,"s3://crabby-images/70271/702713f6d350d1fc7574dacd3889bd30877f758d" alt="在这里插入图片描述"
?
2.4.3 任意位置插入
? index特殊位置分析
data:image/s3,"s3://crabby-images/aa054/aa054046f86eeee24d37c8d616731654b86d0a54" alt="在这里插入图片描述"
? index 正常位置插入 data:image/s3,"s3://crabby-images/899bb/899bb560e3775c1ce7b544e38f68139c90820625" alt="在这里插入图片描述"
代码实现 :
public void addIndex(int index, int data) {
ListNode cur = new ListNode(data);
if (index < 0 || index > size()) {
throw new IndexWrongFulException("index 位置不合法 !!!!");
}
if (index == 0) {
addFirst(data);
return;
}
if (index == size()) {
addLast(data);
return;
}
ListNode cur2 = findIndexSubOne(index);
cur.next = cur2.next;
cur2 = cur;
}
private ListNode findIndexSubOne(int index) {
ListNode cur = head;
while (index - 1 != 0) {
cur = cur.next;
index--;
}
return cur;
}
? 添加就学完了, 会了添加 那么对应的删除 也是不可少的, 下面就来学习一下删除操作 。
?
2.5 删除操作 remove
? 这里主要实现两种 删除 :
- 第一种 删除 :删除第一次出现的 key 元素
- 第二种 删除 :删除链表中 所有的 key 元素
?
2.5.1 删除第一次出现的 key 元素
data:image/s3,"s3://crabby-images/32eb9/32eb9b3598c82bb7e903f6a565433cad66957e1e" alt="在这里插入图片描述"
? 代码实现 :
这里先来完成我们找前一个节点的方法 。
data:image/s3,"s3://crabby-images/5e668/5e6687ce2b32c69d8ad96592d58898694fdcfbd0" alt="在这里插入图片描述"
? 这里的 找前驱 (前一个节点)的方法写完了, 就来完成我们的remove 方法
data:image/s3,"s3://crabby-images/8d68d/8d68dd12d9df35086c5234417d9a14dd7ec68a2b" alt="在这里插入图片描述"
?
2.5.2 删除链表中所有的 key值
? 解法一 (不推荐): 循环调用 删除第一次出现的key 的方法 ? 上面我们写完了删除一个key ,那么删除多个key 还不简单吗? ? 直接 写一个循环 调用 这个方法 , 执行 链表的长度次,这样就能 将 链表中的所有的key 全部删除. ? 为啥不推荐呢 ? 你想我们 删除第一个key 的方法时间复杂度是不是 0(N) , 然后我们有需要循环N 次 (链表长度) ,然后调用这个方法 ,那么时间复杂度是不是就是O(N ^ 2) , 时间复杂度太高了 , 那么 肯定是不推荐的。 ? 下面就来学习一下 时间复杂度O(N) 的 方法。 ?
解法二 :
? 图一 : data:image/s3,"s3://crabby-images/61bf1/61bf110bbd23dbd9a2cad7324c509649c43834fa" alt="在这里插入图片描述"
? 图二 :
data:image/s3,"s3://crabby-images/451fe/451fe91d7155f7e475432919da9acac3a12bf5ad" alt="在这里插入图片描述"
? 图三 :
data:image/s3,"s3://crabby-images/18c17/18c170948d3050531b8332efa2494b885ad4a58f" alt="在这里插入图片描述"
删除方法就完成了, 下面就来完成最后一个方法清空链表
?
2.6 清空操作
? 这里清空操作 可以有两种写法 , 一种比较粗暴一点,直接让头节点置为空 这样就没有人引用 这个链表自然我们就达成了清空操纵 ? 第二种那么就是 ,将每一个节点 的 next 域置为空即可 。 ?
2.6.1 粗暴一点的清空方法
data:image/s3,"s3://crabby-images/b86fc/b86fcf3be49ec3f3f28664871422ddb962c02bb5" alt="在这里插入图片描述"
?
2.6.2 温柔一点的清空方法
data:image/s3,"s3://crabby-images/8bfc1/8bfc1a8eafcb3c8fd06bcc974f6fbee3036b11a7" alt="在这里插入图片描述"
? 附上代码:
import java.util.List;
public class MySingleList {
private int size;
static class ListNode {
public int value;
public ListNode next;
public ListNode(int value) {
this.value = value;
}
}
private ListNode head;
public void create() {
ListNode listNode1 = new ListNode(12);
ListNode listNode2 = new ListNode(23);
ListNode listNode3 = new ListNode(34);
ListNode listNode4 = new ListNode(45);
ListNode listNode5 = new ListNode(56);
listNode1.next = listNode2;
listNode2.next = listNode3;
listNode3.next = listNode4;
listNode5.next = null;
head = listNode1;
}
public void display() {
if (head == null) {
return;
}
ListNode cur = head;
while (cur != null) {
System.out.printf(cur.value + " ");
}
System.out.println();
}
public int size() {
if (head == null) {
return -1;
}
ListNode cur = head;
int count = 0;
while (cur != null) {
count++;
cur = cur.next;
}
return count;
}
public boolean contains(int key) {
ListNode cur = head;
while (cur != null) {
if (cur.value == key) {
return true;
}
cur = cur.next;
}
return false;
}
public void addFirst(int data) {
ListNode cur = new ListNode(data);
cur.next = head;
head = cur;
}
public void addLast(int data) {
ListNode cur = new ListNode(data);
if (head == null) {
head = cur;
return;
}
ListNode cur2 = head;
while (cur2.next != null) {
cur2 = cur2.next;
}
cur2.next = cur;
}
public void addIndex(int index, int data) {
ListNode cur = new ListNode(data);
if (index < 0 || index > size()) {
throw new IndexWrongFulException("index 位置不合法 !!!!");
}
if (index == 0) {
addFirst(data);
return;
}
if (index == size()) {
addLast(data);
return;
}
ListNode cur2 = findIndexSubOne(index);
cur.next = cur2.next;
cur2 = cur;
}
private ListNode findIndexSubOne(int index) {
ListNode cur = head;
while (index - 1 != 0) {
cur = cur.next;
index--;
}
return cur;
}
public void remove(int key) {
if (head == null) {
return;
}
if (head.value == key) {
head = head.next;
return;
}
ListNode cur = findPrevOfKey(key);
if (cur == null) {
System.out.println("没有你要删除的 " + key);
return;
}
cur.next = cur.next.next;
}
public ListNode findPrevOfKey(int key) {
ListNode cur = head;
while (cur.next != null) {
if (cur.next.value == key) {
return cur;
}
cur = cur.next;
}
return null;
}
public void removeAllkey(int key) {
if (head == null) {
return;
}
ListNode prev = head;
ListNode cur = prev.next;
while (cur != null) {
if (cur.value == key) {
prev.next = cur.next;
} else {
prev.next = cur;
}
cur = cur.next;
}
if (head.value == key) {
head = head.next;
}
}
public void clear() {
while(head != null){
ListNode cur = head.next;
head.next = null;
head = cur;
}
}
}
? 这样我们的 单向 不带头不循环的链表就实现完成了, 下面我们就可以来写几道oj题目来看看我们对链表的掌握情况 。 ?
链表OJ题目
链接 提供 :
1.移除链表元素
2.反转链表
3.链表的中间结点
4.链表中倒数第k个结点_
5.合并两个有序链表
6.链表分割
7.链表的回文结构
8.相交链表
9.环形链表
10.环形链表 II
如果上面的题目写完 还 嫌不够的话 ,这里还有 力扣 和牛客 链表的合集 也可以写一些
链表知识点题库 - 力扣(LeetCode)
牛客网 - 链表
本文完 : 下文这些 OJ 题目 的讲解。
|