简介
1、链表(linked List)仍然是有序的列表,且有无头指针根据需求来确定
2、存储方式:节点
节点 = data域(存数据) + next域(指向下一个节点的地址)
3、各个节点不一定连续存放,呈链式结构
地址 | data域 | next域 |
---|---|---|
112 | a1 | 116 |
113 | a2 | 115 |
114 | a3 | 117 |
115 | a4 | 113 |
116 | a5 | 114 |
容易知道,上述表格中链表的逻辑结构:
(head)–a1(116)–a4(117)–a5(114)–a2(115)–a3(117)
链表创建
1、创建头节点head,不存放数据,表示创建的单链表的头
2、所有节点都包含的属性-数据,next:
class HeroNode
int no;
String name;
String nickName;
//前三个为data域
HeroNode next; //next域指向下一个节点
3、最后一个节点的next = null
遍历链表
添加辅助变量来实现遍历
案例1:直接将水浒人物的新数据添加到链表尾部,不考虑排序
定义节点
//首先定义一个HeroNode,每个HeroNode对象就是一个节点
class HeroNode{
public int no;
public String name;
public String nickName;
public HeroNode next; //next域指向下一个节点
//构造器
public HeroNode(int no, String name, String nickName) {
this.no = no;
this.name = name;
this.nickName = nickName;
}
//为了显示方便,重写toString方法
@Override
public String toString() {
return "HeroNode [no=" + no + ", name=" + name + ", nickName=" + nickName + "]";
}
}
定义链表,功能包括1、按主函数的添加顺序添加节点;2、显示链表数据
//定义SingleLinkedList管理英雄人物
class SingleLinkedList{
//初始化一个头节点,不发生变化
private HeroNode head = new HeroNode(0, "", "");
//添加节点到单向链表-从尾部直接添加
//需要找到链表的最后节点,将next指向新节点后,新数节点的next置空
public void addHero(HeroNode heroNode) {
//由于head节点不能动,所以需要一个辅助变量temp来遍历
HeroNode temp = head;
//遍历链表找到最后节点
while(true) {
if(temp.next == null) {
break;
}
temp = temp.next; //没有找到最后就将temp后移
}
temp.next = heroNode; //当退出while循环时,temp保证指向了最后一个节点,此时将数据添加进来
}
//显示链表-遍历
public void list() {
//先判断链表是否为空
if(head.next == null) {
System.out.println("链表为空");
return;
}
//头节点不能变动,引入辅助变量进行遍历
HeroNode temp = head.next;
while(true) {
//判断是否到链表最后(判断是不是该退出循环了)
if(temp == null) {
break; //链表为空
}
System.out.println(temp);
temp = temp.next;
}
}
}
测试:创建节点数据-创建链表对象,将数据按顺序插入链表尾部
public class SingleLinkedListDemo {
public static void main(String[] args) {
//测试
//创建节点
HeroNode heroNode1 = new HeroNode(1, "宋江", "及时雨");
HeroNode heroNode2 = new HeroNode(2, "卢俊义", "玉麒麟");
HeroNode heroNode3 = new HeroNode(3, "吴用", "智多星");
HeroNode heroNode4 = new HeroNode(4, "林冲", "豹子头");
//创建链表
SingleLinkedList singleLinkedList = new SingleLinkedList();
//往链表里插入数据
singleLinkedList.addHero(heroNode1);
singleLinkedList.addHero(heroNode4);
singleLinkedList.addHero(heroNode3);
singleLinkedList.addHero(heroNode2);
singleLinkedList.list();
}
}
由结果可以看出,主函数中的添加顺序如何,显示结果就如何,不能按照no的顺序进行逻辑排序
案例2:根据指定编号顺序添加数据节点
思路
1、首先要找到新数据节点应被添加的位置(如数据1后面)–通过辅助变量temp移动到该位置(数据1),即temp是添加位置的前一个节点(因为单链表只能找下一个节点)
2、让新节点的.next = temp.next
3、再将temp.next = 新节点
代码实现-在Single Linked List中加入以下新方法
//案例2:根据排名添加数据,如果排名重复,则显示添加失败
public void addByOrder(HeroNode heroNode) {
//由于头节点不动,故引入辅助变量进行遍历,找到新数据要添加的数据
HeroNode temp = head;
boolean flag = false; //用于标识新添加的标号是否已经存在,默认为false
while(true) {
if(temp.next==null) {
break;
}
if(temp.next.no > heroNode.no) {
break;
}else if(temp.next.no == heroNode.no){
flag = true; //说明编号存在
break;
}
temp = temp.next;
}
if(flag) {
System.out.println("准备插入的英雄编号"+temp.next.no+"已存在");
}else {
heroNode.next = temp.next;
temp.next = heroNode;
}
}
测试