用(带头节点)单链表完成图书统计。节点结构包括书籍编号,书籍名以及对应作者。功能包括增加(尾插法和指定位置插入法)、删除、修改、查看。
一、定义节点结构
每一个节点都包括这些变量,构造器复杂将传入的数据赋值给类本身的变量。
public static class BookNode{public int bookNumber;public String bookName;public String bookAuthor;public BookNode next;//构造器public BookNode(int no, String book, String author){bookNumber = no;bookName = book;bookAuthor = author;}}
二、定义一个SingleLinkList类
该类用于管理链表,因为使用的是带头节点的单链表,因此在定义类的时候就应该把头节点定义好并初始化。
public static class SingleLinkList{private BookNode head = new BookNode(0," "," ");//......}
三、添加元素
(1)尾插法
入参为一个新建的节点,用temp找到temp.next为null,以为着已经到达了链表尾部。将新节点node赋值给temp.next即已经将新节点连接上链表尾部。
//尾插法public void add1(BookNode node){BookNode temp = head;while(true){if(temp.next == null){break;}temp = temp.next;}temp.next = node;}
测试结果:
因为手动添加的顺序是1-3-2,用尾插法会默认将新节点放在尾部,所以最后输出的时候仍然保持着1-3-2的顺序。
(2)指定位置插入
制定位置插入是根据书本的编号来安排每个节点之间的关系。
何为合适的位置?就是在链表中找到某一个位置,上一个节点的书本编号小于新节点的书本编号,而下一个节点的书本编号又大于新节点的书本编号,这意味着该位置就是新节点的最佳位置。前一个节点就是为temp,而后一个节点就为temp2。若满足条件,则需要断开此处的聊表链接,将新节点插入,再完成链接。
如果寻至尾部,仍未找到满足条件的位置,则在链表最后插入即可。
//中间插入public void add2(BookNode node) {BookNode temp = head;while (true) {if (temp.next == null) {temp.next = node;break;}if (node.bookNumber > temp.bookNumber) {BookNode temp2 = temp.next;if (node.bookNumber < temp2.bookNumber) {BookNode temp3 = temp.next;temp.next = node;node.next = temp3;break;}}temp = temp.next;}}
测试结果:
手动添加的顺序仍然是1-3-2,用制定位置插入法会按照图书标号将新节点放在编号所对应的位置,所以最后输出的顺序为1-2-3。
三、展示链表
首先判断链表是否为空,若为空则提示用户并直接return就可以了;若不为空,则用show遍历链表,并将链表中的数据打印出来即可。
//展示链表public void list(){if(head.next == null){System.out.println("链表为空~没东西可看了......");return;}BookNode show = head.next;while(show != null){System.out.println("(" +show.bookNumber+ ")" +"《"+show.bookName+"》"+"作者:"+show.bookAuthor);show = show.next;}}
四、修改链表
传入一个新的节点,新节点的bookNumber应该与目标节点保持一致。通过对比新节点和目标节点号,寻找要修改的节点。用flag记录是否找到,如找到则将flag设为true,否则false。完成遍历后,根据flag值进行下一步操作。若为true,则意味着在链表中找到了目标节点,把链表中目标节点的数据更新为新节点的数据;若为false,意味着没有找到,则提示用户。
//修改public void updata(BookNode bookdata){BookNode temp = head;boolean flag = false;while(true){if(temp.bookNumber == bookdata.bookNumber){flag = true;break;}temp = temp.next;}if(flag){temp.bookName = bookdata.bookName;temp.bookAuthor = bookdata.bookAuthor;}else{System.out.println("没有找到该元素~");}}
测试结果:
根据新节点的信息修改链表中节点的信息。将第二本图书,更改为无名氏写的《十万个为什么》,但序号不变。
五、删除节点
传入一个需要删除的编号,通过对比每个节点的bookNumber进行删除操作,同样用flag记录找寻结果。当找到需要删除的节点时,temp会指向需要删除的节点,而front则会指向需要删除节点的前一个节点。front变量的存在,就是为了方便删除操作,将front.next的值直接更改为temp.next值(意味着已经跳过temp节点),在java中temp节点则被视为弃用的节点会被自动回收。如果没有找到需要删除的节点,则提示用户。
//删除public void del(int n){BookNode front = head;BookNode temp = head.next;Boolean flag = false;while (true){if(n == temp.bookNumber){flag = true;break;}temp = temp.next;front = front.next;}if(true){front.next = temp.next;}else{System.out.println("没有找到需要删除的数据......");}}
测试结果:
将第一本书鲁迅的《狂人日记》从链表中删除。最后输出仅剩两本书了。
全部代码:
import org.omg.Messaging.SyncScopeHelper;public class LineList {public static void main(String[] args){//书籍数据BookNode test = new BookNode(1,"狂人日记","鲁迅");BookNode test2 = new BookNode(2,"社会性动物","E.阿伦森");BookNode test3 = new BookNode(3,"时间简史","史蒂芬.霍金");//创建一个单链表SingleLinkList testlist = new SingleLinkList();//添加//头插法
// testlist.add1(test);
// testlist.add1(test3);
// testlist.add1(test2);
// //指定位置插入testlist.add2(test);testlist.add2(test3);testlist.add2(test2);// //删除testlist.del(1);
//
// //修改BookNode updata = new BookNode(2,"十万个为什么","无名氏");testlist.updata(updata);//展示链表testlist.list();}//节点public static class BookNode{public int bookNumber;public String bookName;public String bookAuthor;public BookNode next;//构造器public BookNode(int no, String book, String author){bookNumber = no;bookName = book;bookAuthor = author;}}//管理public static class SingleLinkList{private BookNode head = new BookNode(0," "," ");//尾插法public void add1(BookNode node){BookNode temp = head;while(true){if(temp.next == null){break;}temp = temp.next;}temp.next = node;}//中间插入public void add2(BookNode node) {BookNode temp = head;while (true) {if (temp.next == null) {temp.next = node;break;}if (node.bookNumber > temp.bookNumber) {BookNode temp2 = temp.next;if (node.bookNumber < temp2.bookNumber) {BookNode temp3 = temp.next;temp.next = node;node.next = temp3;break;}}temp = temp.next;}}//展示链表public void list(){if(head.next == null){System.out.println("链表为空~没东西可看了......");return;}BookNode show = head.next;while(show != null){System.out.println("(" +show.bookNumber+ ")" +"《"+show.bookName+"》"+"作者:"+show.bookAuthor);show = show.next;}}//修改public void updata(BookNode bookdata){BookNode temp = head;boolean flag = false;while(true){if(temp.bookNumber == bookdata.bookNumber){flag = true;break;}temp = temp.next;}if(flag){temp.bookName = bookdata.bookName;temp.bookAuthor = bookdata.bookAuthor;}else{System.out.println("没有找到该元素~");}}//删除public void del(int n){BookNode front = head;BookNode temp = head.next;Boolean flag = false;while (true){if(n == temp.bookNumber){flag = true;break;}temp = temp.next;front = front.next;}if(true){front.next = temp.next;}else{System.out.println("没有找到需要删除的数据......");}}}
}