目录
一、ArrayList底层源码
首先,我们都知道ArrayList、LinkedList都是Collection接口下的List接口下的实现类,List接口下还有一个Vector实现类,不过已经很少使用Vector了。我们在创建对象时,往往会通过new一个实现类来创建一个集合对象,从而对其进行操作。今天看了一下他们的底层源码,谈谈自己的理解。
一般我们会使用无参构造创建ArrayList对象,此时我们对一个默认容量的空数组进行了初始化,将其初始化为一个数组长度为0的Object类型的数组,同时创建了一个底层用于存取数据的数组elementData。在调用ArrayList的无参构造时,将一个长度为0的数组的地址赋值给elementData。所以一个ArrayList数组在创建时的长度为0。
在ArrayList执行add方法时会调用其私有重载方法。
第一个参数为当前要添加的数据
第二个参数为ArrayList底层数组,要将数据添加到其中
第三个参数为添加元素的索引以及当前数组、ArrayList集合的长度。
对于集合长度与底层数组的长度相等时,我们会对底层数组进行扩容。
由于底层数组已经不能够存下要添加的数据,我们调用了grow的有参方法,底层数组可接受的最小容量便是size+1。
对于原来长度为0的数组,我们会将最小容量与默认容量10进行比较,取最大值对数组进行重新初始化。
对于原来的长度大于0或者底层数组不为空数组的情况下我们调用了newLength方法
我们会将最小添加容量与原容量的0.5倍进行比较,从中取大来获取新数组的容量。然后通过数组工具类Arrays中的CopyOf方法对底层数组进行扩容,将原有的数据存入其中。
总之,
①当我们创建ArrayList对象时,其长度为0
②当我们开始添加元素时,一般情况下会会扩容成一个长度为10的底层数组
③如果容量依旧不够的话,我们会将底层数组扩容至原来长度的1.5倍。
二、LinkedList底层源码
LinkedList<String> list=new LinkedList<>();
list.add("aaa");
list.add("bbb");
list.add("ccc");
首先会调用LinkedList的无参构造,初始化first节点和last节点,分别为双向链表的头结点和尾节点做准备。
首先创建一个l节点暂时存储last的地址,然后创建一个节点,此节点的pre指针指向上一个节点的地址(前一个节点为null的话,指向null),再将新节点的地址赋值给last指针。如果前一个节点为null,直接将此节点变为头节点,反之则会将此节点接在前一个节点l的后面。
总之,LinkedList的底层就是用双向链表实现的,增删快。