【数据结构与算法 | 基础篇】模拟LinkedList实现的链表(无哨兵)

简介: 【数据结构与算法 | 基础篇】模拟LinkedList实现的链表(无哨兵)

1.前言

我们将LinkdList视作链表, 底层设计了内部类Node类, 我这里依然没有用到泛型, 其实加上泛型依然很简单, 即将Node节点的数据域的类型由Int转换为E(<E>), 我在此不做赘述.同时实现了增删查改, 遍历等操作.

2.链表(无哨兵)的代码实现

public class LinkListTest implements Iterable<Integer>{
    //头指针
    static Node head;
    //内部类
    private static class Node{
        //数据域
        int value;
        //指针域
        Node next;
        public Node(int value, Node nest) {
            this.value = value;
            this.next = nest;
        }
    }
    //头插法
    public void addHead(int value) {
        Node p = new Node(value, null);
        p.next = head;
        head = p;
    }
    //遍历链表 : 方式1
    public void Traversal1() {
        Node p = head;
        while (p != null) {
            System.out.println("该节点的值是" + p.value);
            p = p.next;
        }
    }
    //获取指定索引位置上的节点的值
    public int get(int index) {
        Node p = findIndex(index);
        return p.value;
    }
    //返回指定索引的节点
    private Node findIndex(int index) {
        int count = 0;
        Node p = head;
        while (count < index) {
            if (p == null) {
                throw new IllegalArgumentException();
            }
            p = p.next;
            count++;
        }
        return p;
    }
    //找到尾节点
    private Node findLast() {
        //如果是空链表, 压根找不到最后的节点
        if (head == null) {
            return null;
        }
        Node p;
        //当p.next == null时, 则p指向了尾节点
        for (p = head; p.next != null; p = p.next) {
            ;
        }
        return p;
    }
    //尾插法
    public void addLast(int value) {
        Node last = findLast();
        //如果是一个空链表
        if (last == null) {
            addHead(value);
            return;
        }
        last.next = new Node(value, null);
    }
    public void Insert(int index, int value) {
        //如果插入的是第一个节点
        if (index == 0) {
            addHead(value);
            return;
        }
        //先找到index位置处的前一个节点
        Node p = findIndex(index - 1);
        p.next = new Node(value, p.next);
    }
    //删除最后一个节点
    public void removeFrist() {
        //如果是空链表
        if (head == null) {
            throw new RuntimeException();
        }
        //找到头节点
        Node p = findIndex(0);
        head = p.next;
    }
    public void remove(int index) {
        //如果是空链表
        if (head == null) {
            throw new RuntimeException();
        }
        //如果要删除的是最后一个节点, 那么调用removeFrist()方法
        if (index == 0) {
            removeFrist();
            return;
        }
        Node p = findIndex(index - 1);
        p.next = p.next.next;
    }
 
 
    //使用迭代器进行遍历
    @Override
    public Iterator<Integer> iterator() {
        return new Iterator<Integer>() {
            Node q = head;
            @Override
            public boolean hasNext() {
                if (q != null) {
                    return true;
                }
                return false;
            }
 
            @Override
            public Integer next() {
                int value = q.value;
                q = q.next;
                return value;
            }
        };
 
    }
    //使用函数式接口进行链表的遍历
    public void Traverse2(Consumer<Integer> consumer) {
        for (Node p = head; p != null; p = p.next) {
            consumer.accept(p.value);
        }
    }
 
}
 

3. 注

  • foreach循环的底层使用了迭代器.所以如果一个类的对象想要使用foreach循环遍历,则其类必须实现Iterable接口,并重写其中的抽象方法(iterable()).在其return语句中实现了匿名内部类,重写了Iterator接口的两个重要的抽象方法,hasNext()和next().不断迭代将next函数返回的值赋值给临时变量element.
  • 在Traverse2方法中,我们使用了函数式接口来进行链表的遍历
public void Traverse2(Consumer<Integer> consumer) {
        for (Node p = head; p != null; p = p.next) {
            consumer.accept(p.value);
        }
    }
 
@Test
    public void test3() {
        LinkListTest l = new LinkListTest();
        l.addLast(12);
        l.addLast(23);
        l.addLast(34);
        l.addLast(45);
        l.addLast(56);
        l.addLast(67);
        l.Traverse2(value -> {
            System.out.println("该节点的值为" + value);
        });
        l.Traverse2(System.out :: println);
    }

形参是一个消费型的函数式接口,实参我们可以传入接口的实现类的对象(lambda表达式或者方法引用实现).

相关文章
|
2月前
|
存储 算法 Perl
数据结构实验之链表
本实验旨在掌握线性表中元素的前驱、后续概念及链表的建立、插入、删除等算法,并分析时间复杂度,理解链表特点。实验内容包括循环链表应用(约瑟夫回环问题)、删除单链表中重复节点及双向循环链表的设计与实现。通过编程实践,加深对链表数据结构的理解和应用能力。
65 4
|
14天前
|
数据库
数据结构中二叉树,哈希表,顺序表,链表的比较补充
二叉搜索树,哈希表,顺序表,链表的特点的比较
数据结构中二叉树,哈希表,顺序表,链表的比较补充
|
2月前
|
存储 缓存 算法
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式,强调了合理选择数据结构的重要性,并通过案例分析展示了其在实际项目中的应用,旨在帮助读者提升编程能力。
75 5
|
2月前
|
存储 Java 索引
Java中的数据结构:ArrayList和LinkedList的比较
【10月更文挑战第28天】在Java编程世界中,数据结构是构建复杂程序的基石。本文将深入探讨两种常用的数据结构:ArrayList和LinkedList,通过直观的比喻和实例分析,揭示它们各自的优势与局限,帮助你在面对不同的编程挑战时做出明智的选择。
|
2月前
|
存储 C语言
【数据结构】手把手教你单链表(c语言)(附源码)
本文介绍了单链表的基本概念、结构定义及其实现方法。单链表是一种内存地址不连续但逻辑顺序连续的数据结构,每个节点包含数据域和指针域。文章详细讲解了单链表的常见操作,如头插、尾插、头删、尾删、查找、指定位置插入和删除等,并提供了完整的C语言代码示例。通过学习单链表,可以更好地理解数据结构的底层逻辑,提高编程能力。
122 4
|
2月前
|
算法 安全 搜索推荐
2024重生之回溯数据结构与算法系列学习之单双链表精题详解(9)【无论是王道考研人还是IKUN都能包会的;不然别给我家鸽鸽丢脸好嘛?】
数据结构王道第2.3章之IKUN和I原达人之数据结构与算法系列学习x单双链表精题详解、数据结构、C++、排序算法、java、动态规划你个小黑子;这都学不会;能不能不要给我家鸽鸽丢脸啊~除了会黑我家鸽鸽还会干嘛?!!!
|
2月前
|
存储 Web App开发 算法
2024重生之回溯数据结构与算法系列学习之单双链表【无论是王道考研人还是IKUN都能包会的;不然别给我家鸽鸽丢脸好嘛?】
数据结构之单双链表按位、值查找;[前后]插入;删除指定节点;求表长、静态链表等代码及具体思路详解步骤;举例说明、注意点及常见报错问题所对应的解决方法
|
2月前
|
算法
数据结构之购物车系统(链表和栈)
本文介绍了基于链表和栈的购物车系统的设计与实现。该系统通过命令行界面提供商品管理、购物车查看、结算等功能,支持用户便捷地管理购物清单。核心代码定义了商品、购物车商品节点和购物车的数据结构,并实现了添加、删除商品、查看购物车内容及结算等操作。算法分析显示,系统在处理小规模购物车时表现良好,但在大规模购物车操作下可能存在性能瓶颈。
57 0
|
2月前
|
C语言
【数据结构】双向带头循环链表(c语言)(附源码)
本文介绍了双向带头循环链表的概念和实现。双向带头循环链表具有三个关键点:双向、带头和循环。与单链表相比,它的头插、尾插、头删、尾删等操作的时间复杂度均为O(1),提高了运行效率。文章详细讲解了链表的结构定义、方法声明和实现,包括创建新节点、初始化、打印、判断是否为空、插入和删除节点等操作。最后提供了完整的代码示例。
84 0
|
3月前
|
存储 人工智能 算法
数据结构与算法细节篇之最短路径问题:Dijkstra和Floyd算法详细描述,java语言实现。
这篇文章详细介绍了Dijkstra和Floyd算法,这两种算法分别用于解决单源和多源最短路径问题,并且提供了Java语言的实现代码。
105 3
数据结构与算法细节篇之最短路径问题:Dijkstra和Floyd算法详细描述,java语言实现。