【数据结构与算法】2、链表(简单模拟 Java 中的 LinkedList 集合,反转链表面试题)

简介: 【数据结构与算法】2、链表(简单模拟 Java 中的 LinkedList 集合,反转链表面试题)


一、链表基本概念和基本代码实现

🍃 动态数组有个明显的缺点:可能会造成内存空间的大量浪费

🍃 能否用到多少就申请多少内存:链表可以办到

🍃 链表是一种链式存储的线性表,所有元素的内存地址不一定连续


🍀 链表(LinkedList)中有 size 属性【记录着链表中元素的个数、节点的个数】

🍀 链表中的 first 被称作头指针【它引用着链表中的第一个节点(头节点)】

🍀 链表中的元素是存储在Node 节点中的,一个Node 节点可被简单看作是一个元素

🍀 Node 也是一个类

🍀 Node 节点中的 element 属性是该 Node 节点 存储的数据

🍀 Node 节点中的 next 属性引用着下一个节点(next 存储着下一个节点的内存地址)

🍀 尾节点的 next 属性不存储任何内容(① 引用着 null;② 指向 null)

public class LinkedList<E> {
    private int size;
    // 头指针
    private Node<E> first;
    /**
     * 每一个节点就是 Node 类的一个实例
     */
    private static class Node<E> {
        E element;
        Node<E> next;
        public Node(E element, Node<E> next) {
            this.element = element;
            this.next = next;
        }
    }
}

二、链表、动态数组整合(面向接口编程)

🧡 ArrayListLinkedList 继承(extends)抽象类 AbstractList

🧡 抽象类 AbstractList 抽取 ArrayList 和 LinkedList 的公共代码,并实现(implements)了 List 接口

🧡 List 接口定义了 LinkedList 和 ArrayList 要实现的全部代码

三、clear()

🧡 没有被引用的对象会被 Java 的垃圾回收器自动回收

public void clear() {
       size = 0;
       first = null;
   }

四、add(int index, E element)

🌼 要往 index 位置插入元素

🌼 找到 index - 1 位置的节点(记作:prev

🌼 让新节点(记作:newNode)的 next 指向 prev.next(是新节点的下一个节点)

🌼 让 prev 的 next 指针指向新节点(记作:newNode

🌼 size++

(1) 找到 index 位置的节点

☃️ 要找 0 号位置的节点【first 头指针指向的就是 0 号位置的节点】

☃️ 要找 1 号位置的节点【first 头指针的 next 指向的就是 1 号位置的节点】

☃️ 要找 2 号位置的节点【first 头指针的 next 的 next 指向的就是 2 号位置的节点】

☃️ 要找0号位置的节点是0次 next,要找1号位置的节点是1次 next,要找2号位置的节点是2次 next,要找 index 位置的节点是 index 次 next

/**
     * 返回 index 位置的节点
     */
    private Node<E> node(int index) {
        rangeCheck(index);
        // 从头节点开始(默认指向头节点)
        Node<E> moveNode = first;
        for (int i = 0; i < index; i++) {
            moveNode = moveNode.next;
        }
        return moveNode;
    }

(2) get(int index) 和 set(int index, E element)

🌴 因为 node() 方法返回了 index 位置的节点,所以 get(int index) 方法(返回 index 位置的元素 ) 方法也得以实现

🌴 因为 node() 方法返回了 index 位置的节点,所以 set(int index, E element) 方法(设置 index 位置的元素为 element ) 方法也得以实现

@Override
    public E get(int index) {
        Node<E> node = node(index);
        return node.element;
    }
@Override
    public E set(int index, E element) {
        Node<E> node = node(index);
        E old = node.element;
        
        node.element = element;
        return old;
    }

(3) add(int index, E element)

🍀 编写链表代码过程中,要注意边界测试

🍀 比如 index 为 0size – 1size

@Override
    public void add(int index, E element) {
        rangeCheck4Add(index);
        if (index == 0) { // 把元素插入到头节点的位置
            first = new Node<>(element, first);
        } else {
            // 拿到【index - 1】位置的节点
            Node<E> prev = node(index - 1);
            prev.next = new Node<>(element, prev.next);
        }
        size++;
    }

五、remove(int index)

@Override
    public E remove(int index) {
        rangeCheck(index);
        
        Node<E> node = first;
        if (index == 0) { // 删除头节点
            first = node.next;
        } else {
            Node<E> prev = node(index - 1);
            node = prev.next;
            prev.next = node.next;
        }
        size--;
        return node.element;
    }

🌴 注意 index 等于 0 的时候

六、indexOf(E element)

🌴 遍历整个链表,取值比较

@Override
    public int indexOf(E element) {
        Node<E> moveNode = first;
        if (element == null) {
            for (int i = 0; i < size; i++) {
                if (null == moveNode.element) return i;
                moveNode = moveNode.next;
            }
        } else {
            for (int i = 0; i < size; i++) {
                if (element.equals(moveNode.element)) return i;
                moveNode = moveNode.next;
            }
        }
        return ELEMENT_NOT_FOUND;
    }

七、三个 Leetcode 的练习题

(1) 删除链表中的节点

题目地址:https://leetcode.cn/problems/delete-node-in-a-linked-list/

🌼有一个单链表的 head,我们想删除它其中的一个节点 node

🌼 给你一个需要删除的节点 node

🌼 你无法访问第一个节点 head

🌼 链表的所有值都是唯一的

🌼 给定的节点 node 不是链表中的最后一个节点

🌼 删除节点并不是指从内存中删除它

而是:

① 给定节点的值不应该存在于链表中

② 链表中的节点数应该减少 1

node 前面的所有值顺序相同

node 后面的所有值顺序相同



/**
 * Definition for singly-linked list.
 * public class ListNode {
 *     int val;
 *     ListNode next;
 *     ListNode(int x) { val = x; }
 * }
 */
class Solution {
    public void deleteNode(ListNode node) {
        node.val = node.next.val;
        node.next = node.next.next;
    }
}

🍃 用要被删除的节点(delNode)的下一个节点的值(delNode.next.val)覆盖 delNode 的值

🍃 delNode 的 next 指向 delNode 的 next 的 next

(2) 反转一个链表

https://leetcode-cn.com/problems/reverse-linked-list/

① 递归反转一个链表

/**
 * Definition for singly-linked list.
 * public class ListNode {
 *     int val;
 *     ListNode next;
 *     ListNode() {}
 *     ListNode(int val) { this.val = val; }
 *     ListNode(int val, ListNode next) { this.val = val; this.next = next; }
 * }
 */
class Solution {
    public ListNode reverseList(ListNode head) {
        if (head == null || head.next == null) return head;
        ListNode newHead = reverseList(head.next);
        head.next.next = head;
        head.next = null;
        return newHead;
    }
}

② 迭代反转一个链表

/**
 * Definition for singly-linked list.
 * public class ListNode {
 *     int val;
 *     ListNode next;
 *     ListNode() {}
 *     ListNode(int val) { this.val = val; }
 *     ListNode(int val, ListNode next) { this.val = val; this.next = next; }
 * }
 */
class Solution {
    public ListNode reverseList(ListNode head) {
        if (head == null || head.next == null) return head;
        ListNode newHead = null;
        while (head != null) {
            ListNode tmp = head.next;
            head.next = newHead;
            newHead = head;
            head = tmp;
        }
        return newHead;
    }
}

(3) 判断一个链表是否有环(快慢指针)

https://leetcode.cn/problems/linked-list-cycle/

/**
 * Definition for singly-linked list.
 * class ListNode {
 *     int val;
 *     ListNode next;
 *     ListNode(int x) {
 *         val = x;
 *         next = null;
 *     }
 * }
 */
class Solution {
    public boolean hasCycle(ListNode head) {
        if (head == null || head.next == null) return false;
        ListNode slow = head;
        ListNode fast = head.next;
        while (slow != fast) {
            if (fast == null || fast.next == null) return false;
            slow = slow.next;
            fast = fast.next.next;
        }
        return true;
    }
}

完整代码:https://gitee.com/zgq666good/datastructureandalgorithm.git

如有错误请不吝赐教

相关文章
|
9天前
|
存储 安全 Java
Java 集合框架中的老炮与新秀:HashTable 和 HashMap 谁更胜一筹?
嗨,大家好,我是技术伙伴小米。今天通过讲故事的方式,详细介绍 Java 中 HashMap 和 HashTable 的区别。从版本、线程安全、null 值支持、性能及迭代器行为等方面对比,帮助你轻松应对面试中的经典问题。HashMap 更高效灵活,适合单线程或需手动处理线程安全的场景;HashTable 较古老,线程安全但性能不佳。现代项目推荐使用 ConcurrentHashMap。关注我的公众号“软件求生”,获取更多技术干货!
33 3
|
2月前
|
存储 算法 Perl
数据结构实验之链表
本实验旨在掌握线性表中元素的前驱、后续概念及链表的建立、插入、删除等算法,并分析时间复杂度,理解链表特点。实验内容包括循环链表应用(约瑟夫回环问题)、删除单链表中重复节点及双向循环链表的设计与实现。通过编程实践,加深对链表数据结构的理解和应用能力。
65 4
|
14天前
|
数据库
数据结构中二叉树,哈希表,顺序表,链表的比较补充
二叉搜索树,哈希表,顺序表,链表的特点的比较
数据结构中二叉树,哈希表,顺序表,链表的比较补充
|
26天前
|
存储 缓存 安全
Java 集合江湖:底层数据结构的大揭秘!
小米是一位热爱技术分享的程序员,本文详细解析了Java面试中常见的List、Set、Map的区别。不仅介绍了它们的基本特性和实现类,还深入探讨了各自的使用场景和面试技巧,帮助读者更好地理解和应对相关问题。
44 5
|
2月前
|
存储 缓存 安全
Java 集合框架优化:从基础到高级应用
《Java集合框架优化:从基础到高级应用》深入解析Java集合框架的核心原理与优化技巧,涵盖列表、集合、映射等常用数据结构,结合实际案例,指导开发者高效使用和优化Java集合。
47 4
|
2月前
|
存储 缓存 算法
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式,强调了合理选择数据结构的重要性,并通过案例分析展示了其在实际项目中的应用,旨在帮助读者提升编程能力。
76 5
|
2月前
|
算法
数据结构之购物车系统(链表和栈)
本文介绍了基于链表和栈的购物车系统的设计与实现。该系统通过命令行界面提供商品管理、购物车查看、结算等功能,支持用户便捷地管理购物清单。核心代码定义了商品、购物车商品节点和购物车的数据结构,并实现了添加、删除商品、查看购物车内容及结算等操作。算法分析显示,系统在处理小规模购物车时表现良好,但在大规模购物车操作下可能存在性能瓶颈。
57 0
|
存储 Java 程序员
Java面试题日积月累(数据库30道)
Java面试题日积月累(数据库30道)
82 0
|
6月前
|
SQL 安全 Java
Java面试题:什么是JDBC以及如何在Java中使用它进行数据库操作?
Java面试题:什么是JDBC以及如何在Java中使用它进行数据库操作?
65 0
|
6月前
|
druid Java 数据库连接
Java面试题:解释数据库连接池的概念及其作用,讨论常见的连接池实现。
Java面试题:解释数据库连接池的概念及其作用,讨论常见的连接池实现。
102 0
下一篇
开通oss服务