LinkedList

一. LinkedList简介

LinkedList和ArrayList一样, 在集合框架中, LInkedList也是一个类, 实现了List接口:


【说明】

  1. LinkedList 实现了 List 接口
  2. LinkedList 的底层使用了双向链表
  3. LinkedList 没有实现 RandomAccess 接口,因此 LinkedList 不支持随机访问
  4. LinkedList 的任意位置插入和删除元素时效率比较高,时间复杂度为 O(1)
  5. LinkedList 比较适合任意位置插入的场景

但是与我们上次定义的MySingleList不同, 上次的定义的无头单向非循环链表, 而**LinkedLIst是无头双向非循环链表,**画图理解一下:

二. LinkedList的简单实现

无头双向非循环链表的实现:

IList.java:

java 复制代码
public interface IList {
    //头插法
     void addFirst(int data);
    //尾插法
     void addLast(int data);
    //任意位置插入,第一个数据节点为0号下标
     void addIndex(int index,int data);
    //查找是否包含关键字key是否在单链表当中
     boolean contains(int key);
    //删除第一次出现关键字为key的节点
     void remove(int key);
    //删除所有值为key的节点
     void removeAllKey(int key);
    //得到单链表的长度
     int size();
     void clear() ;
     void display() ;

}

IndexException.java:

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

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

KeyException.java:

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

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

MyLinkedList.java:

java 复制代码
/**
 * Created with IntelliJ IDEA.
 * Description:
 * User: Jiang Jinxi
 * Date: 2023-12-27
 * Time: 13:42
 */
public class MyLinkedList implements IList {
    static class ListNode{
        public ListNode prev;
        public int val;
        public  ListNode next;

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

    @Override
    public void addFirst(int data) {
        ListNode node = new ListNode(data);
        if(head == null){
            head = node;
            last = node;
        }else{
            node.next = head;
            head.prev = node;
            head = node;

        }

    }

    @Override
    public void addLast(int data) {
        ListNode node = new ListNode(data);
        if(head == null){
            head = node;
            last = node;
        }else{
            last.next = node;
            node.prev = last;
            last = node;
        }


    }

    @Override
    public void addIndex(int index, int data) {
        if(index < 0 || index > size()){
            throw new IndexException("index 不合法" + index);
        }
        if(index == 0){
            addFirst(data);
            return;
        }
        if(index == size()){
            addLast(data);
            return;
        }
        ListNode cur = head;
        while(index != 0){
            cur = cur.next;
            index--;
        }

//插入结点
       ListNode node = new ListNode(data);
        node.next = cur;
        cur.prev.next = node;
        node.prev = cur.prev;
        cur.prev = node;


    }

    @Override
    public boolean contains(int key) {
        ListNode cur = head;
        while(cur!=null) {
            if (cur.val == key) {
                return true;
            }
            cur = cur.next;
        }
        return false;
    }

    @Override
    public void remove(int key) {
        if(!contains(key)){
            throw new KeyException("找不到key" + key);
        }
//如果只有一个结点且该节点的值就是我要删的数字
       if(head.next == null && head.val == key){
            head = null;
            last = null;
            return;
        }
//循环遍历结点
        ListNode cur = head;
        while(cur!=null) {
            if (cur.val == key) {
//如果要删除的是头结点
                if(cur == head){
                    head = head.next;
                    head.prev = null;
                    return;
                }
//如果要删除的是尾结点
                if(cur == last){
                    last = last.prev;
                    last.next = null;
                    return;
                }
//是中间结点
                cur.prev.next = cur.next;
                cur.next.prev = cur.prev;
                return;

            }
            cur = cur.next;
        }

    }

    @Override
    public void removeAllKey(int key) {
        if(!contains(key)){
            throw new KeyException("找不到key" + key);
        }
        ListNode cur = head;
        if(cur.next == null && cur.val == key){
            head = null;
            last = null;
            return;
        }
        while(cur!=null) {
            if (cur.val == key) {
                if(cur == head){
                    head = head.next;
                    head.prev = null;
                }
                if(cur == last) {
                    last = last.prev;
                    last.next = null;
                }
                cur.prev.next = cur.next;
                cur.next.prev = cur.prev;
            }
            cur = cur.next;
        }

    }

    @Override
    public int size() {
        ListNode cur = head;
        int count = 0;
        while(cur != null){
            cur = cur.next;
            count++;
        }

        return count;
    }

    @Override
    public void clear() {
        head = null;
        last = null;

    }

    @Override
    public void display(){
        ListNode cur = head;
        while(cur != null) {
            System.out.println(cur.val + " ");
            cur = cur.next;//遍历所有节点
        }
    }
}

三. LinkedList的使用

1. LinkList的构造

1)无参构造

// 构造一个空的 LinkedList
List < Integer > list1 = new LinkedList <> ();

2)使用其他集合容器中元素构造List

List < String > list2 = new ArrayList <> ();
list2 . add ( "JavaSE" );
list2 . add ( "JavaWeb" );
list2 . add ( "JavaEE" );
// 使用 ArrayList 构造 LinkedList
List < String > list3 = new LinkedList <> ( list2 );

2. LinkedList****的其他常用方法介绍

方法与ArrayList类似, 不再赘述

3. LinkedList****的遍历

第一种:for循环

// 使用下标+for遍历

for (int i = 0; i < linkedList.size(); i++) {

System.out.print(linkedList.get(i) + " ");

}

第二种: for-each

// 借助foreach遍历

for (Integer x : linkedList) {

System.out.print(x + " ");

}

System.out.println();

第三种: 使用迭代器

可以使用迭代器的原因是: 继承了Iterable接口

从前往后:

Iterator<Integer> it = linkedList.iterator();

while(it.hasNext()){

System.out.print(it.next() + " ");

}

ListIterator<Integer> it = linkedList.listIterator();

while(it.hasNext()){

System.out.print(it.next() + " ");

}

注: ListIterator是Iterator的子类

从后往前:

ListIterator<Integer> it = linkedList.listIterator(linkedList.size() );

while(it.hasPrevious()){

System.out.print(it.previous() + " ");

}

四. ArrayList和LinkedList的区别

相关推荐
跃ZHD几秒前
前后端分离,Jackson,Long精度丢失
java
blammmp21 分钟前
Java:数据结构-枚举
java·开发语言·数据结构
何曾参静谧34 分钟前
「C/C++」C/C++ 指针篇 之 指针运算
c语言·开发语言·c++
暗黑起源喵39 分钟前
设计模式-工厂设计模式
java·开发语言·设计模式
WaaTong44 分钟前
Java反射
java·开发语言·反射
Troc_wangpeng1 小时前
R language 关于二维平面直角坐标系的制作
开发语言·机器学习
努力的家伙是不讨厌的1 小时前
解析json导出csv或者直接入库
开发语言·python·json
Envyᥫᩣ1 小时前
C#语言:从入门到精通
开发语言·c#
九圣残炎1 小时前
【从零开始的LeetCode-算法】1456. 定长子串中元音的最大数目
java·算法·leetcode
wclass-zhengge1 小时前
Netty篇(入门编程)
java·linux·服务器