Java实现单链表以及各种功能

简介: 单链表是一种链式存取的数据结构,用一组地址任意的存储单元存放线性表中的数据元素。

java实现单链表以及 增加、合并、反转等功能

链表的结构很简单,就是一个个节点连接在一起,形成一个完整的链条,每个节点包含2部分,数据域,和一个指向下一个节点引用的指针next。


具体的代码实现

代码里面注释的很清楚了,也有应用于测试的main方法

public class LinkedListDemo {
    public static void main(String[] args) {

        HeroNode hero1 = new HeroNode(1, "林冲", "豹子头");
        HeroNode hero2 = new HeroNode(2, "宋江", "及时雨");
        HeroNode hero3 = new HeroNode(3, "吴用", "智多星");
        HeroNode hero4 = new HeroNode(4, "卢俊义", "玉麒麟");


        LinkedList list = new LinkedList();
        LinkedList list1 = new LinkedList();
        LinkedList list2 = new LinkedList();

        list1.addByOrder(hero1);
        list2.addByOrder(hero4);
        list1.addByOrder(hero3);
        list2.addByOrder(hero2);

        System.out.println("第一个链表中的数据");
        list1.showLinkedList();

        System.out.println("第二个链表中的数据");
        list2.showLinkedList();

        System.out.println("合并后链表中的数据");
        HeroNode combinelist = combineList(list1.head.next, list2.head.next);
        HeroNode print = new HeroNode();
        print = combinelist;
        while (print != null) {
            System.out.println(print);
            print = print.next;
        }

        System.out.println("将合并后的链表倒序输出");
        reverPrint(combinelist);

        System.out.println("将合并后的链表倒转后的链表");
        HeroNode print1 = new HeroNode();
        print1.next = combinelist;
        reversetList(print1);
        while (print1.next != null) {
            System.out.println(print1.next);
            print1 = print1.next;
        }

    }


    /**
     * 查询链表中 有效节点的个数
     * 思路:
     * 1.创建一个临时变量,记录有效节点你的个数
     * 2.便利传进来的 链表
     *
     * @param head 传进来头节点
     * @return 返回有效节点的个数
     */
    private static int indexNode(HeroNode head) {
        //判断链表是否为空
        if (head.next == null) {
            System.out.println("链表为空.");
            return 0;
        }
        //如果链表不为空的话  对链表进行遍历
        int num = 0;
        //创建一个临时变量 用来遍历 操作
        HeroNode term = head.next;
        while (term != null) {
            num++;
            term = term.next;
        }
        return num;
    }

    /**
     * 将节点 加入到链表中 指定节点的位置
     * 思路:
     * 1.首先判断传 进来的链表中 有效数据的 总个数
     * 2.链表中有效数据的总个数 - index(倒数第index个节点) = 得到的结果是 需要返回第几个节点的信息
     *
     * @param head 传进去链表的头部
     * @return 返回 链表中倒数第n个节点的信息
     */
    private static HeroNode findIndexNode(HeroNode head, int index) {
        //首先判断链表是否为空
        if (head.next == null) {
            System.out.println("链表为空。");
            return null;
        }
        HeroNode term = head.next;
        int max = indexNode(head);
        if (max < index) {
            System.out.println("有效数据个数小于  倒数节点的数目。");
            return null;
        }
        for (int i = 0; i < max - index; ++i) {
            term = term.next;
        }

        return term;
    }

    /**
     * 将链表 反转  改变链表的结构
     * 思路:
     * 1.首先创建一个新的头节点
     * 2.让后将传进来的链表的节点信息一个一个从头部插入进去 实现链表的反转
     * 3.最后用传进来的头节点执向新的头节点
     *
     * @param head 传进去待 反转链表的头部
     */
    private static void reversetList(HeroNode head) {
        //判断链表是否回空  或者链表是否只有一个节点
        if (head == null || head.next == null) {
            return;
        }

        //创建新的链表头部
        HeroNode newhead = new HeroNode(0, "", "");
        HeroNode term = head.next;
        HeroNode next = null;

        while (term != null) {
            next = term.next; //保存当前节点的下一个节点的信息

            // 将链表 倒须,  就是将链表的 从第一个节点开始一个一个的将节点插入到链表的 第一个节点的位置
            term.next = newhead.next;   //将当前节点的下一个节点的地址更改为第一个节点的地址(如果当前term节点为第一个节点 那个这个节点操作后就会变成最后一个节点 所以第一个节点的下一个节点的信息为“”)
            newhead.next = term;      //将头节点指向的第一个节点的  信息更改为当前的term节点  实现节点的插入操作
            term = next;

        }
        //让以前的头节点 指向这个新的链表
        head.next = newhead.next;

    }

    /**
     * 反向打印链表  不改变链表的结构
     * 思路
     * 第一种方法:将链表反转 让后打印(不推荐 因为将来链表过大 就太浪费时间了)
     * 第二种方法:利用 栈 先进后出的原则,将链表反向打印
     */
    private static void reverPrint(HeroNode head) {
        //首先判断链表是否为空
        if (head == null) {
            System.out.println("链表为空");
            return;
        }
        //创建一个栈
        Stack<HeroNode> stack = new Stack<>();
        //将 链表中节点的信息 加入到栈中   入栈操作
        HeroNode item = head;
        while (item != null) {
            stack.push(item);
            item = item.next;
        }
        //浆栈 中的元素取出来   出栈操作
        while (stack.size() > 0) {
            System.out.println(stack.pop());
        }

    }

    /**
     * 合并两个已知顺序的链表  按顺序合并
     * 思路:
     * 1.创建一个新的链表
     * 2.将待合并的两个链表 按顺序 比较
     * 3.将小的数据 加入到  新的链表中
     *
     * @param head1 第一个链表头部
     * @param head2 第二个连表头部
     * @return 返回合成后的链表
     */
    private static HeroNode combineList(HeroNode head1, HeroNode head2) {

        if (head1 == null && head2 == null) {
            return null;
        }
        if (head1 == null) {
            return head2;
        } else if (head2 == null) {
            return head1;
        } else {
            HeroNode temp = null;
            if (head1.no > head2.no) {
                temp = head2;
                temp.next = combineList(head1, head2.next);
            } else if (head1.no < head2.no) {
                temp = head1;
                temp.next = combineList(head1.next, head2);
            }
            return temp;
        }
    }


}


class LinkedList {
    //一个空的  链表的头部
    HeroNode head = new HeroNode(0, "", "");

    public void addLinkedList(HeroNode hero) {
        // 思路:  首先寻找链表的尾部  让后将数据插入进去
        //链表的头部不可以动,所以需要一个临时变量,来寻找链表的尾部
        HeroNode term = head;
        //首先遍历整个链表 寻找链表的尾部
        while (true) {
            if (term.next == null) {
                //将数据插入到链表的尾部
                term.next = hero;
                break;
            }
            term = term.next;
        }

    }


    //在指定位置  添加节点
    public void addByOrder(HeroNode hero) {
        HeroNode temp = head;

        while (true) {
            if (temp.next == null) {
                temp.next = hero;
                break;
            }
            //条件满足就把节点 添加到后面
            if (temp.next.no > hero.no) {
                hero.next = temp.next;
                temp.next = hero;
                break;
            } else if (temp.next.no == hero.no) {
                System.out.println("节点" + hero.no + "已存在,添加失败");
                break;
            }
            temp = temp.next;
        }
    }

    //根据 节点的编号进行查找,让后修改
    public void update(HeroNode hero) {
        HeroNode temp = head;

        while (true) {
            if (temp.next == null) {
                System.out.println("链表为空,修改失败!");
                break;
            }
            if (temp.next.no == hero.no) {
                temp.next.name = hero.name;
                temp.next.nickname = hero.nickname;
                break;
            }
            temp = temp.next;
        }

    }

    //删除节点  根据编号进行删除节点,删除后的节点 idea的垃圾回收机制会进行回收
    public void delet(HeroNode hero) {
        HeroNode temp = head;
        while (true) {
            if (temp.next == null) {
                System.out.println("链表为空,删除节点失败。");
                break;
            }
            if (temp.next.no == hero.no) {
                temp.next = temp.next.next;
                break;
            }
            temp = temp.next;
        }
    }

    public void showLinkedList() {

        if (head.next == null) {
            System.out.println("链表为空。");
            return;
        }
        HeroNode term = head.next;
        while (true) {
            System.out.println(term);
            term = term.next;
            if (term == null) {
                break;
            }
        }
    }


}


class HeroNode {
    public int no;
    public String name;
    public String nickname;
    HeroNode next;

    public HeroNode() {
    }

    HeroNode(int no, String name, String nickname) {
        this.no = no;
        this.name = name;
        this.nickname = nickname;
    }

    @Override
    public String toString() {
        return "HeroNode{" +
                "no=" + no +
                ", name='" + name + '\'' +
                ", nickname='" + nickname + '\'' +
                '}';
    }
}

运行结果截图
在这里插入图片描述

目录
相关文章
|
3月前
|
存储 数据可视化 Java
Java Stream API 的强大功能
Java Stream API 是 Java 8 引入的重要特性,它改变了集合数据的处理方式。通过声明式语法,开发者可以更简洁地进行过滤、映射、聚合等操作。Stream API 支持惰性求值和并行处理,提升了代码效率和可读性,是现代 Java 开发不可或缺的工具。
Java Stream API 的强大功能
|
4月前
|
安全 Java API
Java中的Lambda表达式:简洁与功能的结合
Java中的Lambda表达式:简洁与功能的结合
437 211
|
4月前
|
前端开发 JavaScript Java
Java 项目实战城市公园信息管理系统开发流程与实用功能实现指南
本系统基于Java开发,采用Spring Boot后端框架与Vue.js前端框架,结合MySQL数据库,构建了一个高效的城市公园信息管理系统。系统包含管理员、用户和保洁人员三大模块,涵盖用户管理、园区信息查询、订票预约、服务管理等功能,提升公园管理效率与服务质量。
162 6
|
4月前
|
安全 Java 数据库
Java 项目实战病人挂号系统网站设计开发步骤及核心功能实现指南
本文介绍了基于Java的病人挂号系统网站的技术方案与应用实例,涵盖SSM与Spring Boot框架选型、数据库设计、功能模块划分及安全机制实现。系统支持患者在线注册、登录、挂号与预约,管理员可进行医院信息与排班管理。通过实际案例展示系统开发流程与核心代码实现,为Java Web医疗项目开发提供参考。
218 2
|
4月前
|
机器学习/深度学习 算法 Java
Java 大视界 -- Java 大数据机器学习模型在生物信息学基因功能预测中的优化与应用(223)
本文探讨了Java大数据与机器学习模型在生物信息学中基因功能预测的优化与应用。通过高效的数据处理能力和智能算法,提升基因功能预测的准确性与效率,助力医学与农业发展。
|
4月前
|
JavaScript Java 微服务
现代化 Java Web 在线商城项目技术方案与实战开发流程及核心功能实现详解
本项目基于Spring Boot 3与Vue 3构建现代化在线商城系统,采用微服务架构,整合Spring Cloud、Redis、MySQL等技术,涵盖用户认证、商品管理、购物车功能,并支持Docker容器化部署与Kubernetes编排。提供完整CI/CD流程,助力高效开发与扩展。
502 64
|
5月前
|
Java API
深入解析Java API中Object类的功能
了解和合理运用 Object类的这些方法,对于编写可靠和高效的Java应用程序至关重要。它们构成了Java对象行为的基础,影响着对象的创建、识别、表达和并发控制。
106 0
|
5月前
|
消息中间件 监控 Java
借助最新技术构建 Java 邮件发送功能的详细流程与核心要点分享 Java 邮件发送功能
本文介绍了如何使用Spring Boot 3、Jakarta Mail、MailHog及响应式编程技术构建高效的Java邮件发送系统,涵盖环境搭建、异步发送、模板渲染、测试与生产配置,以及性能优化方案,助你实现现代化邮件功能。
212 0
|
5月前
|
算法 安全 Java
java中Collections.shuffle方法的功能说明
`Collections.shuffle()` 是 Java 中用于随机打乱列表顺序的方法,基于 Fisher-Yates 算法实现,常用于洗牌、抽奖等场景。可选 `Random` 参数支持固定种子以实现可重复的随机顺序。方法直接修改原列表,无返回值。
185 0
|
5月前
|
Java API
Java API中Math类功能全景扫描
在实际使用时,这些方法的精确度和性能得到了良好的优化。当处理复杂数学运算或高精度计算时,`Math`类通常是足够的。然而,对于非常精细或特殊的数学运算,可能需要考虑使用 `java.math`包中的 `BigDecimal`类或其他专业的数学库。
135 11