目录
单向链表哨兵
单向链表里面有一个特殊的节点称为哨兵节点,不存储数据。
优势:简化了单向链表的空判断,例如 尾插、get、insert、remove
初始
public class SentinelLinkedListTest {
// 头指针 指向哨兵(666是任意定义的一个值)
private Node head = new Node(666, null);
// 节点类 private对外隐藏细节
@Data
@AllArgsConstructor
private static class Node {
private int value;
private Node next;
}
}
将内部类定义为静态主要有两个原因:
实例化方式:静态内部类的实例化不需要依赖于外部类。而普通的内部类在实例化时会隐含地包含一个对外部类的引用,因此,普通的非静态内部类不能脱离外部类实例而单独存在。
访问方式:静态内部类可以使用静态方法直接通过类名来访问外部类的静态成员,而不需要创建外部类的实例。而普通的内部类需要先创建外部类的实例,然后通过该实例来访问外部类的静态成员。
头插
思路
对于使用者来说,我给你一个value值,你要将我的这个value值放入到链表头结点处。
不同于单向链表,这里一开始就有了哨兵
代码
public void addFirst(int value) {
head.next = new Node(value, head.next);
}
哨兵的next指针指向新节点,新节点的next指针为之前哨兵的next指针指向的节点
尾插
思路
对于使用者来说,我给你一个value值,你要将我的这个value值放入到链表最后面
那我怎么知道你链表什么时候是最后面?
遍历
遍历到最后一个节点,此时它的next指针指向null
注意:头指针是为了记录第一个节点地址,不能动,所以我定义一个可以移动的指针开始指向第一个节点
public void foreach(Consumer<Integer> consumer) {
Node p = head.next;
while (p != null) {
consumer.accept(p.value);
p = p.next;
}
}
注:不同于单向链表,自定义的指针开始是指向哨兵的下一个节点地址。