单链表
什么是单链表?
单链表是一种链式存取的数据结构,用一组地址任意的存储单元存放线性表中的数据元素。链表中的数据是以结点来表示的,每个结点的构成:元素(val)+ 指针(指示后继元素存储位置),元素就是存储数据的存储单元,指针就是连接每个结点的地址数据。
头指针head和终端结点:
单链表中每个结点的存储地址是存放在其前趋结点next域中,而开始结点无前趋,故应设头指针head指向开始结点。链表由头指针唯一确定,单链表可以用头指针的名字来命名。
终端结点无后继,故终端结点的指针域为空,即NULL。
单链表的创建与添加操作:
(注意:这里我没有用泛型,使用的是int 类型)
在这个单链表中,我写了2个关于添加的方法:
1.add方法,当我们添加一个节点时,它会按照顺序添加节点
2.addByVal方法:当我们添加一个节点时,它会根据这个节点的val值进行排序(按照节点的val从大到小进行排序)
//自己写的单链表
public class MyselfLinkedList {
//先初始化一个头结点,不存放具体的数据
final Node head = new Node(0);
/*
添加节点到单链表中
思路:
1.找到当前节点的最后节点
2.将最后这个节点的next指向新的节点
*/
public void add(Node node) {
//因为head节点不能动,因此我们需要一个辅助节点
Node temp = head;
//遍历链表,找到最后的节点
while (true) {
if (temp.next == null) {
break;
} else {
//如果没有找到最后的节点,将temp后移
temp = temp.next;
}
}
//当退出while循环时,temp就指向了链表最后的节点
//将这个节点的最后指向新的节点
temp.next = node;
}
//第二种方式添加节点到单链表中(根据val的大小添加节点到指定位置)
public void addByVal(Node node) {
//因为head节点不能动,因此我们需要一个辅助节点(指针)来帮助找到添加的位置;
//因为单链表,因此我们找的temp 是位于添加位置的前一个节点,否则插入不了;
Node temp = head;
boolean flag = false;//标志添加的val是否已经存在,默认为false;
while (true) {
if (temp.next == null) {//说明temp已经在链表的最后
break;
}
if (temp.next.val > node.val) {//说明位置已经找到,就在temp的后面插入
break;
} else if (temp.next.val == node.val) {//说明添加的节点的val已经存在;
flag = true;
break;
}
temp = temp.next;//后移,遍历当前链表
}
if (flag) {//说明添加的节点的val已经存在;我们可以让它不添加,也可以让它添加,这里我选择不添加,让它输出一句话
System.out.println("准备插入的节点的val为" + node.val + "已经存在,不能加入\n");
} else {
//插入到链表中,temp的后面
node.next = temp.next;
temp.next = node;
}
}
//显示链表,遍历链表
public void showList() {
//先判断链表是否为空
if (head.next == null) {
System.out.println("链表为空");
} else {
Node temp = head.next;
while (true) {
if (temp == null) {
break;
} else {
System.out.println(temp);
//将temp后移
temp = temp.next;
}
}
}
}
}
在这里我写了两个main方法分别对add和addByVal这两个方法进行测试:
1.add方法的测试:
public static void main(String[] args) {
MyselfLinkedList myselfLinkedList = new MyselfLinkedList();
//先创建节点
Node node = new Node(1);
Node node1 = new Node(2);
Node node2 = new Node(1);
Node node3 = new Node(3);
Node node4 = new Node(7);
Node node5 = new Node(6);
Node node6 = new Node(4);
myselfLinkedList.add(node);
myselfLinkedList.add(node1);
myselfLinkedList.add(node2);
myselfLinkedList.add(node3);
myselfLinkedList.add(node4);
myselfLinkedList.add(node5);
myselfLinkedList.add(node6);
myselfLinkedList.showList();
/*myselfLinkedList.addByVal(node);
myselfLinkedList.addByVal(node1);
myselfLinkedList.addByVal(node2);
myselfLinkedList.addByVal(node3);
myselfLinkedList.addByVal(node4);
myselfLinkedList.addByVal(node4);*/
}
}
运行结果如下图所示:
我们可以看到节点是我们按照添加的顺序来排序的.
2.addByVal方法的测试
public static void main(String[] args) {
MyselfLinkedList myselfLinkedList = new MyselfLinkedList();
//先创建节点
Node node = new Node(1);
Node node1 = new Node(2);
Node node2 = new Node(1);
Node node3 = new Node(3);
Node node4 = new Node(7);
Node node5 = new Node(6);
Node node6 = new Node(4);
/*myselfLinkedList.add(node);
myselfLinkedList.add(node1);
myselfLinkedList.add(node2);
myselfLinkedList.add(node3);
myselfLinkedList.add(node4);
myselfLinkedList.add(node5);
myselfLinkedList.add(node6);*/
myselfLinkedList.addByVal(node);
myselfLinkedList.addByVal(node1);
myselfLinkedList.addByVal(node2);
myselfLinkedList.addByVal(node3);
myselfLinkedList.addByVal(node4);
myselfLinkedList.addByVal(node5);
myselfLinkedList.addByVal(node6);
myselfLinkedList.showList();
}
}
运行结果如下图所示:
在这里我们可以看到节点的添加是按照节点的val值来添加的,由于我们在addByVal方法中对节点的val值相同的没有进行添加处理,只是输出了一句话,所以由于添加的节点node和节点node2的val值一样都为1,因此该单链表的长度比添加的节点个数少了一个,还在控制台打印了一句话.