模拟实现单链表 —— SingleLinkedList

模拟实现 java 中单链表的实现,方便后续对 java 中的 LInkedList 进行理解。

MySingleList类:

java 复制代码
public class MySingleList {

    /**
     * 定义节点类
     */
    static class ListNode {
        // 节点值
        private int val; 
        // 下一个节点的引用
        private ListNode next; 

        public ListNode(int val) {
            this.val = val;
        }
    }

    // 头节点
    private ListNode head;

    /**
     * 头插法
     */
    public void headInsert(int value) {
        // 申请节点
        ListNode node = new ListNode(value);

        // 头插
        node.next = head;
        head = node;
    }

    /**
     * 尾插法
     */
    public void tailInsert(int value) {
        // 申请节点
        ListNode node = new ListNode(value);

        // head 为空
        if (head == null) {
            // head 直接指向 node
            head = node;
            return;
        }

        ListNode cur = head;
        // 找最后一个节点
        while (cur.next != null) {
            cur = cur.next;
        }

        cur.next = node;
    }

    /**
     * 任意位置插入
     */
    public void randomInsert(int index, int value) {
        // 判断下标是否合理
        judge(index);

        // 判断是否是头插或尾插
        if (isInsert(index, value)) {
            return;
        }

        // 中间位置插入
        ListNode cur = head;
        // 找到插入位置的前一个位置
        for (int i = 0; i < index - 1; i++) {
            cur = cur.next;
        }

        // 插入
        ListNode node = new ListNode(value);
        node.next = cur.next;
        cur.next = node;

    }

    /**
     * 判断下标是否合理
     */
    private void judge(int index) {
        if (index < 0 || index > size()) {
            throw new IndexOFBoundException("下标: " + index + "错误!");
        }
    }

    /**
     * 判断插入位置是否是头插或尾插
     */
    private boolean isInsert(int index, int value) {
        if (index == 0) { // 头插
            headInsert(value);
            return true;
        } else if (index == size()) { // 尾插
            tailInsert(value);
            return true;
        } else {
            return false;
        }
    }

    /**
     * 查找 value 是否在链表中存在
     */
    public boolean contains(int value) {
        ListNode cur = head;

        while (cur != null) {
            if (cur.val == value) {
                return true;
            }
            cur = cur.next;
        }
        return false;
    }

    /**
     * 删除链表中第一次出现 value 的节点
     */
    public void remove(int value) {
        // head 为空或要删除的节点是 head
        if (isNullOrHead(value)) {
            return;
        }

        // 获取要删除节点的前一个节点
        ListNode preNode = findFirstDeletePreNode(value);
        // 判断 preNode 为空
        if (preNode == null) {
            System.out.println("没有这个元素对应的节点");
            return;
        }

        // 删除节点
        ListNode delNode = preNode.next;
        preNode.next = delNode.next;
    }

    /**
     * 要删除的节点为空或是第一个节点
     */
    private boolean isNullOrHead(int value) {
        if (head == null) { // head 为空
            return true;
        } else if (head.val == value) { // head 的 val 和 value 相同
            // head 后移
            head = head.next;
            return true;
        } else {
            return false;
        }
    }

    /**
     * 查找要删除 value 节点
     */
    private ListNode findFirstDeletePreNode(int value) {
        ListNode cur = head;
        while (cur.next != null) {
            // 找到了要删除节点的前驱
            if (cur.next.val == value) {
                return cur;
            }
            cur = cur.next;
        }
        // 没找到
        return null;
    }

    /**
     * 删除所有值为 value 的节点
     */
    public void removeAllValue(int value) {
        // 判空
        if (head == null) return;

        ListNode prev = head;
        ListNode cur = head.next;

        while (cur != null) {
            // 判断 cur.val 是否和 value 相等
            if (cur.val == value) {
                // 相等则让 prev.next 指向 cur.next
                prev.next = cur.next;
            } else {
                // prev 移动到 cur 的位置
                prev = cur;
            }
            // cur 移动到 cur.next
            cur = cur.next;
        }
        // 判断第一个节点的值是否和 value 相等
        if (head.val == value) {
            head = head.next;
        }
    }

    /**
     * 获取单链表的长度
     */
    public int size() {
        ListNode cur = head;
        int count = 0;

        while (cur != null) {
            count++;
            cur = cur.next;
        }
        return count;
    }

    /**
     * 清空单链表
     */
    public void clear() {
        // 把 head 置为 null
        this.head = null;
    }

    /**
     * 打印数组中的元素
     */
    public void display() {
        ListNode cur = head;

        while (cur != null) {
            System.out.print(cur.val + " ");
            // 指向下一个值
            cur = cur.next;
        }
        System.out.println();
    }
}

IndexOFBoundException类:

java 复制代码
public class IndexOFBoundException extends RuntimeException {
    public IndexOFBoundException() {
    }

    public IndexOFBoundException(String message) {
        super(message);
    }
}

Test类

java 复制代码
public class Test {
    public static void main(String[] args) {
        MySingleList mySingleList = new MySingleList();
        /*mySingleList.headInsert(1);
        mySingleList.headInsert(2);
        mySingleList.headInsert(3);*/

        //mySingleList.display();

        mySingleList.tailInsert(1);
        mySingleList.tailInsert(2);
        mySingleList.tailInsert(3);
        mySingleList.tailInsert(4);
        mySingleList.tailInsert(5);
        mySingleList.tailInsert(6);
        mySingleList.display();

        mySingleList.randomInsert(0, 111);
        mySingleList.randomInsert(7, 111);
        mySingleList.randomInsert(3, 111);
        //mySingleList.randomInsert(12, 111);
        mySingleList.display();


        mySingleList.remove(2);
        mySingleList.display();

        mySingleList.removeAllValue(111);
        mySingleList.display();

    }
}
相关推荐
九圣残炎18 分钟前
【从零开始的LeetCode-算法】3227. 字符串元音游戏
java·算法·leetcode
梁小憨憨44 分钟前
变分推断(Variational Inference)
人工智能·算法·机器学习
就爱学编程1 小时前
重生之我在异世界学编程之C语言:选择结构与循环结构篇
c语言·数据结构·算法
一只大侠1 小时前
输入一串字符,以“?”结束。统计其中字母个数,数字个数,其它符号个数。:JAVA
java·开发语言·算法
winstongit1 小时前
捷联惯导原理和算法预备知识
算法·机器人
想成为高手4992 小时前
深入理解AVL树:结构、旋转及C++实现
开发语言·数据结构·c++
£suPerpanda2 小时前
P3916 图的遍历(Tarjan缩点和反向建边)
数据结构·c++·算法·深度优先·图论
IT古董2 小时前
【机器学习】机器学习的基本分类-监督学习-决策树-C4.5 算法
人工智能·学习·算法·决策树·机器学习·分类
m0_694938012 小时前
Leetcode打卡:棋盘上有效移动组合的数目
算法·leetcode·职场和发展
kitesxian2 小时前
Leetcode543. 二叉树的直径(HOT100)
算法·深度优先