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

功能点有:

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. VMware Workstation中安装Hyper-V

    1:在虚拟机设置中,CPU属性中勾选"Virtualize Intel VT-x/EPT or AMD-V/RVI"来启用虚拟机的CPU支持虚拟化. 2:2.在虚拟机文件所在目录中 ...

  2. OO第三单元作业(JML)总结

    OO第三单元作业(JML)总结 目录 OO第三单元作业(JML)总结 JML语言知识梳理 使用jml的目的 jml注释结构 jml表达式 方法规格 类型规格 SMT Solver 部署JMLUnitN ...

  3. Spring Cloud Gateway 之获取请求体(Request Body)的几种方式

    Spring Cloud Gateway 获取请求体 一.直接在全局拦截器中获取,伪代码如下 private String resolveBodyFromRequest(ServerHttpReque ...

  4. mybatis-plus批量插入saveBatch太慢?我愿意称rewriteBatchedStatements为神

    最近在做项目优化,代码优化之后,测试接口,好家伙.一个定时任务接口执行要10秒左右. 一点点追踪,给每个方法打上执行时间,一点点缩小范围.好家伙,终于让我锁定了目标. 这是mybatis-plus的批 ...

  5. Spring Cloud Alibaba Nacos Config 实战

    Nacos 提供用于存储配置和其他元数据的 key/value 存储,为分布式系统中的外部化配置提供服务器端和客户端支持.使用 Spring Cloud Alibaba Nacos Config,您可 ...

  6. 使用CSS样式的三种方法

    一.内联样式 内联样式通过style属性来设置,属性值可以任意的CSS样式. 1 <!DOCTYPE html> 2 <html lang="en"> 3 ...

  7. [刷题] 112 Path Sum

    要求 给出一个二叉树及数字sum,判断是否存在一条从根到叶子的路径,路径上的所有节点和为sum 实现 转化为寻找左右子树上和为 sum-root 的路径,到达叶子节点时递归终止 注意只有一个孩子时,根 ...

  8. Shell $?获取函数返回值或者上一个命令的退出状态

    Shell $?获取函数返回值或者上一个命令的退出状态 来自:互联网 时间:2021-01-12 阅读:2 $? 是一个特殊变量,用来获取上一个命令的退出状态,或者上一个函数的返回值. 所谓退出状态, ...

  9. 3.13eval函数

    eval 函数 eval() 函数十分强大 -- 将字符串 当成 有效的表达式 来求值 并 返回计算结果 ```python 基本的数学计算 In [1]: eval("1 + 1" ...

  10. C语言编程 菜鸟练习100题(31-40)

    [练习31]判断质数 0. 题目: 判断质数 1. 分析: 质数(prime number),指大于 1的.且除 1 和本身以外没有其他因数的自然数. 2. 程序: #include <stdi ...