什么也不说, 直接上代码:

功能点有:

1, 获取尾结点

2, 添加(添加节点到链表的最后面)

3, 添加(根据节点的no(排名)的大小, 有序添加)

4, 单向链表的 遍历

5, 链表的长度

6, 删除某一个节点

7, 更换指定位置的节点

8, 查询第n个节点

9, 查询倒数第n个节点

10, 链表反转, 使用递归实现

11, 逆序打印

12, 合并二个有序链表, 且结果仍然是有序的

//英雄节点
class HeroNodeLv{
public int no;//英雄排名
public String name;//名字
public String nickName;//花名
public HeroNodeLv next;//指向下一节点 public HeroNodeLv(int no, String name, String nickName) {
this.no = no;
this.name = name;
this.nickName = nickName;
} @Override
public String toString() {
return "HeroNodeLv{" +
"no=" + no +
", name='" + name + '\'' +
", nickName='" + nickName + '\'' +
'}';
}
}

//链表对象

    class SingleLinkListLvcai {
//单链表是由节点组成, 节点由 数据+ next(指向下一节点)组成
// 定义一个头结点,为空
public HeroNodeLv head = new HeroNodeLv(0,"","");
public void setHead(HeroNodeLv head) {
this.head = head;
} // 获取头结点
public HeroNodeLv getHeadNode(){return head;} //1, 获取尾结点
public HeroNodeLv getLastNode(){
//head头结点不能动, 定义第三方对象来遍历
HeroNodeLv temp = head;
while(true){
if(temp.next == null){
//说明是尾结点
return temp;
}
temp = temp.next;
}
} //2, 添加(添加节点到链表的最后面)
public void add(HeroNodeLv node){
//head头结点不能动, 定义第三方对象来遍历
HeroNodeLv temp = head;
while(true){
if(temp.next == null){
//说明是尾结点
break;
}
temp = temp.next;
}
//遍历完后,此时temp就是尾结点
temp.next = node;
} //3, 添加(根据节点的no(排名)的大小, 有序添加)
public void addByHeroNo(HeroNodeLv node){
HeroNodeLv temp = head;
boolean flag = false; //该节点是否已存在标识, 默认false,不存在
while(true){
if (temp.next == null) {
//说明到尾节点
break;
}
if(temp.next.no > node.no){
//说明 node 是需要放在 temp的下一个节点
break;
}else if(temp.next.no == node.no){
flag = true;//该节点已存在
break;
}
temp = temp.next;
} if(false){
System.out.println("该节点已存在");
}else{
//插在temp的后面, 这里先将node的next指定好,之后,再指定插入, 这二行代码顺序不能变,否则会出现死循环
node.next = temp.next;
temp.next = node;
}
} //4, 单向链表的 遍历
public void show(){
if(head.next== null){
//空链表
System.out.println("空链表");
return;
}
HeroNodeLv temp = head;
while(true){
if(temp.next == null){
break;
}
System.out.println(temp);
temp = temp.next;
}
System.out.println(temp);
} //5, 链表的长度
public int length(){
if(head.next == null){
return 0;//这里头节点 不计入长度
}
HeroNodeLv temp = head.next;//这里头节点 不计入长度
int count = 0;
while(temp!= null){
count++;
temp = temp.next;
}
return count;
}
//6, 删除某一个节点
public void delNode(int no){
HeroNodeLv temp = head;
boolean flag = false;//该链表是否存在这个节点,默认为false, 不存在
while(true){
if(temp.next == null){
break;//链表尾部
}
if(temp.next.no == no){
flag = true;
break;//删除此时的这个节点的下一节点
}
temp = temp.next;
}
//循环完后,,此时temp的 next就是要删除的节点
if(flag){
temp.next = temp.next.next;
}else {
System.out.println("链表中不存在这个节点");
}
}
//7, 更换指定位置的节点
public void updateNode(int no,HeroNodeLv node){
HeroNodeLv temp = head;
boolean flag = false;//该链表是否存在这个节点, 默认为false 不存在
while (true) {
if (temp.next == null) {
break;//链表尾部
}
if (temp.next.no == no) {
flag = true; //temp的next就是要替换的节点
break;
}
temp = temp.next;
} if (flag) {
//更换
temp.next = node;//更换next.
node.next = temp.next.next;//指定node的next
}else{
System.out.println("不存在这个节点");
}
} //8, 查询第n个节点
public HeroNodeLv getByNo(int no){
//获取链表长度
if(no == 0){
return getHeadNode();//头结点
}
if(no >length()){
System.out.println("超出链表的长度:"+length());
return null;
}
HeroNodeLv temp = head.next;
int count = 0;
while(temp != null){
count++;
if(count == no){
//这就是要找的节点
break;
}
temp = temp.next;
}
return temp;
} //9, 查询倒数第n个节点
public HeroNodeLv getByBackwordNum(int no){
//思路: 查询倒数第n个节点, 就是查询正数第 length-n+1 个节点
//如果为0 或者超出链表长度, 返回null
if(no == 0 || no > length()){
System.out.println("不能为0,或者,超出链表长度:链表长度为"+length());
return null;
}
return getByNo(length() - no +1);
}
} //10, 链表反转, 使用递归实现
public static void reverseList(HeroNodeLv node){
//思路: 遍历原来的链表,每遍历一个节点,就将其取出,并放在新的链表reverseHead 的最前端
if(node.next == null || node.next.next == null){
return ;//空链表,只有一个节点的链表
}
HeroNodeLv cur = node.next;//定义指针变量,便于遍历原来的链表
HeroNodeLv next = null;//next 指的是当前节点(cur)的下一个节点
HeroNodeLv reverseHead = new HeroNodeLv(0,"","");
while(cur!= null){
next = cur.next;//先保存当前节点的写一个节点
cur.next = reverseHead.next;//将cur的下一个节点 指向新链表的最前端
reverseHead.next = cur;//将cur连接到新的链表上
cur = next;//后移
}
//循环完之后,newNode就是一个反转的链表了,,此时将newNode链表的头节点 , 变为原来链表的头节点
node.next = reverseHead.next;
} //11, 逆序打印
public static void reversePrint(HeroNodeLv node){
//思路: 将单链表遍历,一个个取出,放到栈中,然后栈遍历, 利用栈的特点:先进后出
if(node.next == null){
return;
}
HeroNodeLv cur = node.next;
Stack<HeroNodeLv> stack = new Stack<HeroNodeLv>();
while(cur != null){
stack.push(cur);
cur = cur.next;
}
//遍历stack
while(stack.size() > 0){
System.out.println(stack.pop());
}
} //12, 合并二个有序链表, 且结果仍然是有序的, 此方法的入参节点,从第一个节点开始, 不带头节点
public static HeroNodeLv mergeLinkList(HeroNodeLv node1 ,HeroNodeLv node2){
if(node1 == null && node2 == null){
return null;
}
if(node1 == null){
return node2;
} if (node2 == null) {
return node1;
}
HeroNodeLv head = null ;
if(node1.no > node2.no){
head = node2;
head.next = mergeLinkList(node1,node2.next);
}else if(node1.no < node2.no){
head = node1;
head.next = mergeLinkList(node1.next,node2);
}else{
//相等的情况
head = node1;
head.next = mergeLinkList(node1.next,node2.next);
}
return head;
}

测试结果:

合并链表:

         public static void main(String[] args) {
HeroNodeLv node1 = new HeroNodeLv(1, "宋江", "及时雨");
HeroNodeLv node2 = new HeroNodeLv(2, "吴用", "神算子");
HeroNodeLv node3 = new HeroNodeLv(3, "卢俊义", "玉麒麟");
HeroNodeLv node4 = new HeroNodeLv(4, "武松", "打老虎");
HeroNodeLv node5 = new HeroNodeLv(5, "吕财", "打老虎");
HeroNodeLv node6 = new HeroNodeLv(6, "吴二娘", "打老虎");
HeroNodeLv node7 = new HeroNodeLv(7, "吴二娘", "打老虎");
HeroNodeLv node8 = new HeroNodeLv(8, "吴二娘", "打老虎");
HeroNodeLv node9 = new HeroNodeLv(9, "吴二娘", "打老虎");
HeroNodeLv node10 = new HeroNodeLv(10, "吴二娘", "打老虎"); SingleLinkListLvcai linkListLv = new SingleLinkListLvcai();
linkListLv.add(node1);
linkListLv.add(node7);
linkListLv.add(node8);
linkListLv.addByHeroNo(node3);//这里的添加,是有序添加
linkListLv.addByHeroNo(node9);
System.out.println("链表linkListLv:");
linkListLv.show();// 1-3-7-8-9
System.out.println("=============================================="); SingleLinkListLvcai linkListLv2 = new SingleLinkListLvcai();
linkListLv2.addByHeroNo(node7);
linkListLv2.addByHeroNo(node4);
linkListLv2.addByHeroNo(node2);
linkListLv2.addByHeroNo(node8);
linkListLv2.addByHeroNo(node6);
linkListLv2.addByHeroNo(node5);
System.out.println("链表linkListLv2:");
linkListLv2.show();//2-4-5-6-7-8
System.out.println("=============================================="); HeroNodeLv heroNodeLv = mergeLinkList(linkListLv.getHeadNode().next, linkListLv2.getHeadNode().next);
//合并后的链表没有头结点,所以需要加上头结点
HeroNodeLv newLinkList = new HeroNodeLv(0, "", "");
newLinkList.next= heroNodeLv;
linkListLv.setHead(newLinkList);
System.out.println("合并后的新链表为:newLinkList");
linkListLv.show();//1-2-3-4-5-6-7-8
}

合并二个有序链表结果:



链表反转结果:



链表反转打印的结果,不破坏原有链表结构:



获取链表倒数第n个节点:



其余的几个功能点可以自己测试......

2、java数据结构和算法:单链表: 反转,逆序打印, 合并二个有序链表,获取倒数第n个节点, 链表的有序插入的更多相关文章

  1. C++面试常见问题——04链表的逆序与合并

    链表的逆序与合并 链表的逆序 已知一个链表的头指针为head,将该链表逆序. #include<iostream> using namespace std; struct Node{ in ...

  2. [剑指offer] 14. 链表中倒数第K个节点+翻转+逆序打印+合并两个排序链表 + 链表相交(第一个公共节点) (链表)

    题目描述 输入一个链表,输出该链表中倒数第k个结点. 思路:  两个指针,起始位置都是从链表头开始,第一个比第二个先走K个节点,当第一个走到链表尾时,第二个指针的位置就是倒数第k个节点.(两指针始终相 ...

  3. Java实现单链表的逆序打印

    思路1:可以将链表进行反转,然后进行数据的输出即可,单链表反转地址如下https://blog.csdn.net/Kevinnsm/article/details/113763272 这个思路1肯定有 ...

  4. 6, java数据结构和算法: 栈的应用, 逆波兰计算器, 中缀表达式--> 后缀表达式

    直接上代码: public class PolandCalculator { //栈的应用:波兰计算器: 即: 输入一个字符串,来计算结果, 比如 1+((2+3)×4)-5 结果为16 public ...

  5. Java数据结构和算法(一)线性结构之单链表

    Java数据结构和算法(一)线性结构之单链表 prev current next -------------- -------------- -------------- | value | next ...

  6. Java数据结构和算法(四)--链表

    日常开发中,数组和集合使用的很多,而数组的无序插入和删除效率都是偏低的,这点在学习ArrayList源码的时候就知道了,因为需要把要 插入索引后面的所以元素全部后移一位. 而本文会详细讲解链表,可以解 ...

  7. Java数据结构和算法(一)线性结构

    Java数据结构和算法(一)线性结构 数据结构与算法目录(https://www.cnblogs.com/binarylei/p/10115867.html) 线性表 是一种逻辑结构,相同数据类型的 ...

  8. 【Java数据结构学习笔记之二】Java数据结构与算法之栈(Stack)实现

      本篇是java数据结构与算法的第2篇,从本篇开始我们将来了解栈的设计与实现,以下是本篇的相关知识点: 栈的抽象数据类型 顺序栈的设计与实现 链式栈的设计与实现 栈的应用 栈的抽象数据类型   栈是 ...

  9. java数据结构与算法之栈(Stack)设计与实现

    本篇是java数据结构与算法的第4篇,从本篇开始我们将来了解栈的设计与实现,以下是本篇的相关知识点: 栈的抽象数据类型 顺序栈的设计与实现 链式栈的设计与实现 栈的应用 栈的抽象数据类型 栈是一种用于 ...

随机推荐

  1. hdu5007 小水题

    题意:       给你一个串,如果出现子串 "Apple", "iPhone", "iPod", "iPad"输出MA ...

  2. LAMP环境搭建一个Discuz论坛

    LAMP是Linux+Apache+Mysql/MariaDB+Perl/PHP/Python的简称.一组常用来搭建动态网站或者服务器的开源软件,本身都是各自独立的程序,但是因为常被放在一起使用,拥有 ...

  3. hdu2492 数状数组或者线段树

    题意:      给你一些人,每个人有自己的攻击力,输入的顺序就是每个人的顺序,他们之间互相比赛,两个人比赛的条件是必须在他们两个位置之间找到一个人当裁判,这个裁判的攻击力必须在他们两个人之间,问你最 ...

  4. Python中数据的排序

    目录 列表的排序 sort(key,reverse)方法 sorted(target,key,reverse) 函数 元组tuple的排序 sort(key,reverse)方法 sorted(tar ...

  5. Windows Pe 第三章 PE头文件(上)

    第三章  PE头文件 本章是全书重点,所以要好好理解,概念比较多,但是非常重要. PE头文件记录了PE文件中所有的数据的组织方式,它类似于一本书的目录,通过目录我们可以快速定位到某个具体的章节:通过P ...

  6. Day009 Arrays类

    Arrays类 数组的工具类java.util.Arrays 由于数组对象本身并没有什么方法可以供我们调用,但Api中提供了一个工具类Arrays供我们使用,从而可以对数据对象进行一些基本的操作. 查 ...

  7. Idea一直卡在loading archetype list问题解决(或者报Unable to import maven project: See logs for details)

    暂时没有测试成功 https://blog.csdn.net/calo_missile/article/details/95898519

  8. Java发送邮件报错:com.sun.mail.util.LineOutputStream.<init>(Ljava/io/OutputStream;Z)V

    在练习使用Java程序发送邮件的代码 运行出现了com.sun.mail.util.LineOutputStream.<init>(Ljava/io/OutputStream;Z)V报错信 ...

  9. 2021/5/11 docker的应用

    很快一天过去了,今天虽然没有加班,但是依旧感觉疲惫,弄了一天的docker镜像的问题.作为一名前端开发工程师从以前从未听说过docker到现在懂得了如何运用,也是不容易啊.之前也说过,我们项目结构是前 ...

  10. XD to Flutter 2.0 现已发布!

    Flutter 是 Google 的开源 UI 工具包.利用它,只需一套代码库,就能开发出适合移动设备.桌面设备.嵌入式设备以及 web 等多个平台的精美应用.过去几年,对于想要打造多平台应用的开发者 ...