#Java LinkedList集合之链表模拟

闲来无事简单的模拟下双向链表的实现,就当为自己巩固知识和加深理解了,很多人说背不住八股文,那是很正常的,因为没有体验过底层的实现不能理解其原理,所以在干巴巴的背八股文时就很容易将其淡忘。

链表基本介绍

在集合中我们常见的链表就是LinkedList,它是一条双向链表,即每个节点都保存了上下节点的数据,可以让我们对其进行正反向的遍历。

链表模拟

废话不多说,我们开始模拟,看过LinkedList底层都知道,在使用LinkedList时并不是将数据直接存放在LinkedList里面,而是使用其内部类Node对数据进行了封装(源码展示):

// LinkedList的内部节点类
private static class Node<E> {
    E item; // 存放的元素本体
    Node<E> next; // 该元素所在节点的下一个节点
    Node<E> prev; // 该元素所在节点的上一个节点
    
    // 节点的构造器
    Node(Node<E> prev, E element, Node<E> next) {
        this.item = element;
        this.next = next;
        this.prev = prev;
    }
}

那么我们要做的就是先将链表的整体结构先实现出来(这里我自己实现的类名后添加了Sim进行区分)

// LinkedList链表模拟类
public class LinkedListSim {
    // 链表的首节点
    NodeSim first;
    // 链表的尾结点
    NodeSim last;
    // 记录链表的长度
    int size;

    // Node节点模拟类(省略了setter/getter的实现)
    private static class NodeSim { 
        Object date; // 元素本体
        NodeSim next; // 上一个节点
        NodeSim prev; // 下一个节点

        @Override
        public String toString() {
            return "NodeSim{" +
                    "prev=" + prev +
                    ", next=" + next +
                    ", data=" + data +
                    '}';
        }
    }
}

一个简单的双向链表模拟类就完成了,其实模拟链表的结构并不复杂,但如果说只有一个链表的结构那自然是不够的,不能使用就等于没有价值

增删改查

增加

为了让链表变得灵活起来,我们需要让它具备基本的CRUD操作,首先来实现它的add()方法

/**
 * 添加元素
 * @param data 要添加的元素
 */
public void add(Object data) {
    // 将数据封装为Node节点
    NodeSim node = new NodeSim();
    node.setData(data);
    // 若first为null表示添加第一个元素
    if (first == null) {
        // 因为是第一个元素,所以首尾节点都是它,且它没有上下节点
        node.setPrev(null);
        node.setNext(null);
        first = node;
        last = node;
    } else {
        // 否则表示已添加过元素
        node.setPrev(last); // 当前尾节点为该新元素的上一个节点
        node.setNext(null); // 该节点为末尾节点所以没有下一个节点
        // 当前尾结点的下一个节点变更为该节点
        last.setNext(node);
        // 当前尾结点变更为新的尾结点
        last = node;
    }
    size ++;
}

实现add()方法后,我们使用debug进行测试,看看是否能成功存入元素,我们在此处打上断点,通过debug界面查看首节点是否为"a",尾结点是否为"c"

 当断点向下走一步后,也就是所有数据被添加完成,我们发现链表中首尾节点保存的数据都如愿以偿的正确,保险起见我们打开首节点的next节点和尾结点的prev节点,看看是否为中间的元素"b"

 如图所示,结果正如我们所期待的那样,每个存放元素的节点都正确的保存了它们的上下节点

                

 查找

我们熟知LinkedList和ArrayList查找的方式完全不同,这取决于他们底层实现的原理,ArrayList底层使用数组,所以特点就是支持随机下标访问,而LinkedList底层实现为链表,链表是不支持下标访问的,可java还是为我们提供了LinkedList的下标查找,这是为什么呢?

 原因其实很简单,LinkedList的下标查找并不是真正的通过下标进行查找,而是从first首节点一个一个的遍历当前节点的next节点,直到遍历到index后就获取到目标结果

(因为链表除了first和last,它并不能直接知道该节点在链表中的位置,只能从first或last开始数)

首尾节点

 好了,说完原生的LinkedList,那么我们来进行实现自己链表的查找类api,我们知道LinkedList链表中有两个关键的属性first和last,它们分别直接保存的整条链表的首尾节点,因此不论这条链表有多长,当我们只想要获取首尾节点时,它的效率和耗时都是一样(可以直接获取所以非常快)

/**
 * 获取首节点元素
 * @return 首节点元素
 */
public Object getFirst() {
    return first.getData();
}

/**
 * 获取尾结点元素
 * @return 尾结点元素
 */
public Object getLast() {
    return last.getData();
}

链表长度

查看集合内的元素数量(长度),也是必须的功能,但这对链表来说极为简单,只需要返回属性中的size即可

/**
 * 获取链表长度
 * @return 链表长度
 */
public int getSize() {
    return size;
}

下标查找

一个集合我们不可能只查看它的长度,也不可能只查找它的首尾节点,所以我们需要它支持下标查找(并不是真的下标)

/**
 * 获取元素 通过下标
 * @param index 该元素下标
 * @return 该元素
 */
public Object get(int index) {
    // 若查找下标为0说明是查找首节点,直接返回首节点
    if (index == 0) {
        return getFirst();
    } else if (index == (size - 1)) {
        //若查找下标为链表总长-1说明是查找尾结点,直接返回尾结点
        return getLast();
    }

    // 获取链表头元素
    NodeSim node = first;
    for (int i = 0; i < index; i++) {
        node = node.getNext(); // 取该节点的下一个节点
    }
    // 遍历到index个next即获取到目标元素
    return node.getData();
}

测试

实现完几个关键的api后我们进行统一的测试,看看是否能获取正确的结果

(为了方便我就不debug了,直接打印)

测试数据

LinkedListSim link = new LinkedListSim();
    link.add("a");
    link.add("b");
    link.add("c");
    link.add("d");
    link.add("e");
    link.add("f");

test1

获取链表长度getSize()以及获取链表首尾节点getFirst()/getLast()

System.out.println("链表长度:" + link.getSize());
System.out.println("首元素:" + link.getFirst());
System.out.println("尾元素:" + link.getLast());

结果展示

 test2

获取链表中指定的某个index元素

System.out.println("获取下标为0的元素:" + link.get(0)); // 其实就相当于获取首节点
System.out.println("获取下标为3的元素:" + link.get(3));

我们知道index=0表示集合的第一位元素(在链表中为首节点),在上面的get()方法实现中已经进行了优化,当我们在查找时,若index等于首尾节点那么就直接返回首尾节点即可

 模拟完成

在此次LinkedList链表模拟中,我进行了简单的增和查操作实现,看完后你有没有受到一些启发?如果有疑问可以将你的不解留在评论中,若你有所感悟可以试着挑战自己将更改和删除操作进行实现并测试,看看能否独自顺利完成

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值