之前学习了顺序表,接下来把链表的功能给模拟实现一遍
链表
链表是一种物理存储结构上非连续存储结构,数据元素的逻辑顺序是通过链表中的引用链接次序实现的 。
data:image/s3,"s3://crabby-images/06f6d/06f6dc785685399c56b4feccdfab0e50a722c55e" alt=""
链表的结构有很多种,但是我们重点掌握两种:
无头单向非循环链表:结构简单,一般不会单独用来存数据。实际中更多是作为其他数据结构的子结构,如哈希桶、图的邻接表等等。另外这种结构在笔试面试中出现很多。
data:image/s3,"s3://crabby-images/57c2b/57c2b292c886e1d7826b297984b16622db4d39fa" alt=""
整体结构就长这个样子
无头双向链表:在Java的集合框架库中LinkedList底层实现就是无头双向循环链表。
data:image/s3,"s3://crabby-images/38197/3819785981403368fd03ca47fe09cde3fc7eabdd" alt=""
链表的实现
data:image/s3,"s3://crabby-images/574c3/574c3e181d36953e7debc36d7fb02baa839f0a5c" alt=""
第一个节点也称为头结点 head
依靠head 节点就可以找到所有的节点
单链表的模拟实现
data:image/s3,"s3://crabby-images/79f82/79f823934dbc86655cbf0cf861929637aa3f23c0" alt=""
creatList为我们已经创建好了一个链表,在它的基础上我们可以进行操作
data:image/s3,"s3://crabby-images/0c148/0c148aae7a9e756c4ce688d88a16d6906681c670" alt=""
data:image/s3,"s3://crabby-images/2cd5e/2cd5e502fdd8c074fdf7e750be392a44c9feff3b" alt=""
实现接口的功能
data:image/s3,"s3://crabby-images/51ad5/51ad5ddab2acba38bdcea7dab0e7f93e5a25751a" alt=""
一共实现的功能就这么多 现在我们先来一一实现
一.打印链表
data:image/s3,"s3://crabby-images/21899/218991f883e0466a0129e38ea38496231b00e8c4" alt=""
注:一般情况不动head那个节点,创建一个cur节点来代替head节点,让它永远指向头结点
java
public void display() {
if(head == null) {
return ;
}
ListNode cur = head;
while(cur != null) {
System.out.print(cur.val + " ");
cur = cur.next;
}
}
data:image/s3,"s3://crabby-images/53195/53195443b915b46b98200ceb3580e1ad9d868c35" alt=""
二. 查找是否包含关键字key是否在单链表当中
java
public boolean contains(int key) {
ListNode cur = head;
while(cur != null) {
if(cur.val == key) {
return true;
}
cur = cur.next;
}
return false;
}
三.得到单链表的长度
java
public int size() {
int count = 0;
ListNode cur = head;
while(cur != null) {
count++;
cur = cur.next;
}
return count;
}
data:image/s3,"s3://crabby-images/0d50e/0d50e769866f8842acc0ecae44a8e8d1434c09ce" alt=""
这几个都是遍历整个链表,所有会比较轻松
四.头插法
思路:
考虑head等于null的情况下,直接让node = head 就行 这题考虑与不考虑都无所谓
java
public void addFirst(int data) {
ListNode node = new ListNode(data);
if(head == null) {
head = node;
} else {
node.next = head;
head = node;
}
}
data:image/s3,"s3://crabby-images/07892/07892e183310d3970e065c9b2ba47f9836b9a89e" alt=""
五.尾插法
data:image/s3,"s3://crabby-images/80d80/80d8041a7e563796594373668b0984018937c0ed" alt=""
也是一样的 如果head等于null的话 直接让head = node
java
public void addLast(int data) {
ListNode node = new ListNode(data);
ListNode cur = head;
if(head == null) {
head = node;
} else {
while(cur.next != null) {
cur = cur.next;
}
cur.next = node;
}
}
data:image/s3,"s3://crabby-images/305bc/305bc92b4600399a40116ac9e9398cddb9030947" alt=""
六.任意位置插入,第一个数据节点为0号下标
data:image/s3,"s3://crabby-images/8da16/8da16c08e46ff08ebf06880e8700594b05b2d821" alt=""
index = 0 或者等于size()大小时候,就可以用到头插和尾差,再者看看index值合法不合法
java
public void addIndex(int index, int data) {
ListNode node = new ListNode(data);
if(index < 0 || index >= size()) {
throw new InderException("index不合法");
}
if(index == 0) {
addFirst(data);
return;
}
if(index == size()) {
addLast(data);
return;
}
else {
ListNode cur = head;
ListNode curPre = cur;
while( index != 0) {
curPre = cur;
cur = cur.next;
index--;
}
curPre.next = node;
node.next = cur;
}
}
data:image/s3,"s3://crabby-images/7a973/7a973bcfe89cc055d0849340f597b12f7a7a9598" alt=""
七. 删除第一次出现关键字为key的节点
data:image/s3,"s3://crabby-images/19dd7/19dd7c44c9211c4d99050e56c3b0a0cd13205bc4" alt=""
万一第一个节点就是我们要删除的,直接将head 节点移动到下一个节点
java
public void remove(int key) {
if(head == null) {
return;
}
if(head.val == key) {
head = head.next;
return;
}
ListNode node = new ListNode(key);
ListNode cur = head.next;
ListNode curPre = head;
while(cur != null) {
if(cur.val == key) {
curPre.next = cur.next;
return;
}
curPre = curPre.next;
cur = cur.next;
}
}
data:image/s3,"s3://crabby-images/c671f/c671f5243241ef5fec4b608c5c62a8c4e8fa46df" alt=""
八.删除所有值为key的节点
data:image/s3,"s3://crabby-images/acd49/acd49542c4683235478ce9cc43477dd8dddde865" alt=""
头结点也是最后一个判断,只有这种,才可以避免头结点是要删除的节点的问题
java
public void removeAllKey(int key) {
if(head == null) {
return ;
}
ListNode curPre = head;
ListNode cur = head.next;
while(cur != null) {
if(cur.val == key) {
curPre.next = cur.next;
cur = cur.next;
}
else {
cur = cur.next;
curPre = curPre.next;
}
}
if(head.val == key) {
head = head.next;
}
}
data:image/s3,"s3://crabby-images/785a3/785a302a5701129526b03e71ec406809df6c2173" alt=""
八.清空链表
java
public void clear() {
head = null;
}
直接让head 为空就行
ok以上就是整个单链表的模拟过程,这里只是简单入个门而已
单链表一般在笔试面试题常常出现,所有我们要经常刷题才能更好的掌握它
这里为大家整理好了一些题目
以前写的 之前因为个人原因断了 现在慢慢更新 感谢!!!