前言

在一开始基础面的时候,很多面试官可能会问List集合一些基础知识,比如:

  • ArrayList默认大小是多少,是如何扩容的?

  • ArrayListLinkedList的底层数据结构是什么?

  • ArrayListLinkedList的区别?分别用在什么场景?

  • 为什么说ArrayList查询快而增删慢?

  • Arrays.asList方法后的List可以扩容吗?

  • modCount在非线程安全集合中的作用?

  • ArrayListLinkedList的区别、优缺点以及应用场景


ArrayList(1.8)

ArrayList是由动态再分配的Object[]数组作为底层结构,可设置null值,是非线程安全的。

ArrayList成员属性

  1. //默认的空的数组,在构造方法初始化一个空数组的时候使用
  2. private static final Object[] EMPTY_ELEMENTDATA = {};

  3. //使用默认size大小的空数组实例
  4. private static final Object[] DEFAULTCAPACITY_EMPTY_ELEMENTDATA = {};

  5. //ArrayList底层存储数据就是通过数组的形式,ArrayList长度就是数组的长度。
  6. transient Object[] elementData;

  7. //arrayList的大小
  8. private int size;

那么ArrayList底层数据结构是什么呢?

很明显,使用动态再分配的Object[]数组作为ArrayList底层数据结构了,既然是使用数组实现的,那么数组特点就能说明为什么ArrayList查询快而增删慢?

因为数组是根据下标查询不需要比较,查询方式为:首地址+(元素长度*下标),基于这个位置读取相应的字节数就可以了,所以非常快;但是增删会带来元素的移动,增加数据会向后移动,删除数据会向前移动,导致其效率比较低。

ArrayList的构造方法

  • 带有初始化容量的构造方法

  • 无参构造方法

  • 参数为Collection类型的构造器

  1. //带有初始化容量的构造方法
  2. public ArrayList(int initialCapacity) {
  3. //参数大于0,elementData初始化为initialCapacity大小的数组
  4. if (initialCapacity > 0) {
  5. this.elementData = new Object[initialCapacity];
  6. //参数小于0,elementData初始化为空数组
  7. } else if (initialCapacity == 0) {
  8. this.elementData = EMPTY_ELEMENTDATA;
  9. //参数小于0,抛出异常
  10. } else {
  11. throw new IllegalArgumentException("Illegal Capacity: "+
  12. initialCapacity);
  13. }
  14. }

  15. //无参构造方法
  16. public ArrayList() {
  17. //在1.7以后的版本,先构造方法中将elementData初始化为空数组DEFAULTCAPACITY_EMPTY_ELEMENTDATA
  18. //当调用add方法添加第一个元素的时候,会进行扩容,扩容至大小为DEFAULT_CAPACITY=10
  19. this.elementData = DEFAULTCAPACITY_EMPTY_ELEMENTDATA;
  20. }

那么ArrayList默认大小是多少?

从无参构造方法中可以看出,一开始默认为一个空的实例elementData为上面的DEFAULTCAPACITY_EMPTY_ELEMENTDATA,当添加第一个元素的时候会进行扩容,扩容大小就是上面的默认容量DEFAULT_CAPACITY10

ArrayList的Add方法

  • boolean add(E):默认直接在末尾添加元素

  • void add(int,E):在特定位置添加元素,也就是插入元素

  • boolean addAll(Collection<? extends E> c):添加集合

  • boolean addAll(int index, Collection<? extends E> c):在指定位置后添加集合

boolean add(E)
  1. public boolean add(E e) {
  2. ensureCapacityInternal(size + 1); // Increments modCount!!
  3. elementData[size++] = e;
  4. return true;
  5. }

通过ensureCapacityInternal方法为确定容量大小方法。在添加元素之前需要确定数组是否能容纳下,size是数组中元素个数,添加一个元素size+1。然后再数组末尾添加元素。

其中,ensureCapacityInternal方法包含了ArrayList扩容机制grow方法,当前容量无法容纳下数据时1.5倍扩容,进行:

  1. private void ensureCapacityInternal(int minCapacity) {
  2. //判断当前的数组是否为默认设置的空数据,是否取出最小容量
  3. if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {
  4. minCapacity = Math.max(DEFAULT_CAPACITY, minCapacity);
  5. }
  6. //包括扩容机制grow方法
  7. ensureExplicitCapacity(minCapacity);
  8. }

  9. private void ensureExplicitCapacity(int minCapacity) {
  10. //记录着集合的修改次数,也就每次add或者remove它的值都会加1
  11. modCount++;

  12. //当前容量容纳不下数据时(下标超过时),ArrayList扩容机制:扩容原来的1.5倍
  13. if (minCapacity - elementData.length > 0)
  14. grow(minCapacity);
  15. }

  16. private void grow(int minCapacity) {
  17. // overflow-conscious code
  18. int oldCapacity = elementData.length;
  19. //ArrayList扩容机制:扩容原来的1.5倍
  20. int newCapacity = oldCapacity + (oldCapacity >> 1);
  21. if (newCapacity - minCapacity < 0)
  22. newCapacity = minCapacity;
  23. if (newCapacity - MAX_ARRAY_SIZE > 0)
  24. newCapacity = hugeCapacity(minCapacity);
  25. // minCapacity is usually close to size, so this is a win:
  26. elementData = Arrays.copyOf(elementData, newCapacity);
  27. }

ArrayList是如何扩容的?

根据当前的容量容纳不下新增数据时,ArrayList会调用grow进行扩容:

  1. //相当于int newCapacity = oldCapacity + oldCapacity/2
  2. int newCapacity = oldCapacity + (oldCapacity >> 1);

扩容原来的1.5倍。

void add(int,E)
  1. public void add(int index, E element) {
  2. //检查index也就是插入的位置是否合理,是否存在数组越界
  3. rangeCheckForAdd(index);
  4. //机制和boolean add(E)方法一样
  5. ensureCapacityInternal(size + 1); // Increments modCount!!
  6. System.arraycopy(elementData, index, elementData, index + 1,
  7. size - index);
  8. elementData[index] = element;
  9. size++;
  10. }

ArrayList的删除方法

  • remove(int):通过删除指定位置上的元素,

  • remove(Object):根据元素进行删除,

  • clear():elementData中每个元素都赋值为null,等待垃圾回收将这个给回收掉,

  • removeAll(collection c):批量删除。

remove(int)
  1. public E remove(int index) {
  2. //检查下标是否超出数组长度,造成数组越界
  3. rangeCheck(index);

  4. modCount++;
  5. E oldValue = elementData(index);
  6. //算出数组需要移动的元素数量
  7. int numMoved = size - index - 1;
  8. if (numMoved > 0)
  9. //数组数据迁移,这样会导致删除数据时,效率会慢
  10. System.arraycopy(elementData, index+1, elementData, index,
  11. numMoved);
  12. //将--size上的位置赋值为null,让gc(垃圾回收机制)更快的回收它。
  13. elementData[--size] = null; // clear to let GC do its work
  14. //返回删除的元素
  15. return oldValue;
  16. }

为什么说ArrayList删除元素效率低?

因为删除数据需要将数据后面的元素数据迁移到新增位置的后面,这样导致性能下降很多,效率低。

remove(Object)
  1. public boolean remove(Object o) {
  2. //如果需要删除数据为null时,会让数据重新排序,将null数据迁移到数组尾端
  3. if (o == null) {
  4. for (int index = 0; index < size; index++)
  5. if (elementData[index] == null) {
  6. //删除数据,并迁移数据
  7. fastRemove(index);
  8. return true;
  9. }
  10. } else {
  11. //循环删除数组中object对象的值,也需要数据迁移
  12. for (int index = 0; index < size; index++)
  13. if (o.equals(elementData[index])) {
  14. fastRemove(index);
  15. return true;
  16. }
  17. }
  18. return false;
  19. }

可以看出,arrayList是可以存放null值。


LinkedList(1.8)

LinkedList是一个继承于AbstractSequentialList的双向链表。它也可以被当做堆栈、队列或双端队列进行使用,而且LinkedList也为非线程安全, jdk1.6使用的是一个带有 header节头结点的双向循环链表, 头结点不存储实际数据 ,在1.6之后,就变更使用两个节点firstlast指向首尾节点。

LinkedList的主要属性

  1. //链表节点的个数
  2. transient int size = 0;
  3. //链表首节点
  4. transient Node<E> first;
  5. //链表尾节点
  6. transient Node<E> last;
  7. //Node节点内部类定义
  8. private static class Node<E> {
  9. E item;
  10. Node<E> next;
  11. Node<E> prev;

  12. Node(Node<E> prev, E element, Node<E> next) {
  13. this.item = element;
  14. this.next = next;
  15. this.prev = prev;
  16. }
  17. }

一旦变量被transient修饰,变量将不再是对象持久化的一部分,该变量内容在序列化后无法获得访问

LinkedList构造方法

无参构造函数, 默认构造方法声明也不做,firstlast节点会被默认初始化为null。

  1. *
  2. /** Constructs an empty list. \*/*

  3. public LinkedList() {}

LinkedList插入

由于LinkedList由双向链表作为底层数据结构,因此其插入无非由三大种

  • 尾插: add(E e)addLast(E e)addAll(Collection<? extends E> c)

  • 头插: addFirst(E e)

  • 中插: add(int index, E element)

可以从源码看出,在链表首尾添加元素很高效,在中间添加元素比较低效,首先要找到插入位置的节点,在修改前后节点的指针。

尾插-add(E e)和addLast(E e)
  1. //常用的添加元素方法
  2. public boolean add(E e) {
  3. //使用尾插法
  4. linkLast(e);
  5. return true;
  6. }

  7. //在链表尾部添加元素
  8. public void addLast(E e) {
  9. linkLast(e);
  10. }

  11. //在链表尾端添加元素
  12. void linkLast(E e) {
  13. //尾节点
  14. final Node<E> l = last;
  15. final Node<E> newNode = new Node<>(l, e, null);
  16. last = newNode;
  17. //判断是否是第一个添加的元素
  18. //如果是将新节点赋值给last
  19. //如果不是把原首节点的prev设置为新节点
  20. if (l == null)
  21. first = newNode;
  22. else
  23. l.next = newNode;
  24. size++;
  25. //将集合修改次数加1
  26. modCount++;
  27. }
头插-addFirst(E e)
  1. public void addFirst(E e) {
  2. //在链表头插入指定元素
  3. linkFirst(e);
  4. }

  5. private void linkFirst(E e) {
  6. //获取头部元素,首节点
  7. final Node<E> f = first;
  8. final Node<E> newNode = new Node<>(null, e, f);
  9. first = newNode;
  10. //链表头部为空,(也就是链表为空)
  11. //插入元素为首节点元素
  12. // 否则就更新原来的头元素的prev为新元素的地址引用
  13. if (f == null)
  14. last = newNode;
  15. else
  16. f.prev = newNode;
  17. //
  18. size++;
  19. modCount++;
  20. }
中插-add(int index, E element)

index不为首尾的的时候,实际就在链表中间插入元素。

  1. // 作用:在指定位置添加元素
  2. public void add(int index, E element) {
  3. // 检查插入位置的索引的合理性
  4. checkPositionIndex(index);

  5. if (index == size)
  6. // 插入的情况是尾部插入的情况:调用linkLast()。
  7. linkLast(element);
  8. else
  9. // 插入的情况是非尾部插入的情况(中间插入):linkBefore
  10. linkBefore(element, node(index));
  11. }

  12. private void checkPositionIndex(int index) {
  13. if (!isPositionIndex(index))
  14. throw new IndexOutOfBoundsException(outOfBoundsMsg(index));
  15. }

  16. private boolean isPositionIndex(int index) {
  17. return index >= 0 && index <= size;
  18. }

  19. void linkBefore(E e, Node<E> succ) {
  20. // assert succ != null;
  21. final Node<E> pred = succ.prev; // 得到插入位置元素的前继节点
  22. final Node<E> newNode = new Node<>(pred, e, succ); // 创建新节点,其前继节点是succ的前节点,后接点是succ节点
  23. succ.prev = newNode; // 更新插入位置(succ)的前置节点为新节点
  24. if (pred == null)
  25. // 如果pred为null,说明该节点插入在头节点之前,要重置first头节点
  26. first = newNode;
  27. else
  28. // 如果pred不为null,那么直接将pred的后继指针指向newNode即可
  29. pred.next = newNode;
  30. size++;
  31. modCount++;
  32. }

LinkedList 删除

删除和插入一样,其实本质也是只有三大种方式,

  • 删除首节点:removeFirst()

  • 删除尾节点:removeLast()

  • 删除中间节点 :remove(Object o)remove(int index)

在首尾节点删除很高效,删除中间元素比较低效要先找到节点位置,再修改前后指针指引。

删除中间节点-remove(int index)和remove(Object o)
  1. remove(int index)和remove(Object o)都是使用删除指定节点的unlink删除元素
  2.  
  3. public boolean remove(Object o) {
  4. //因为LinkedList允许存在null,所以需要进行null判断
  5. if (o == null) {
  6. //从首节点开始遍历
  7. for (Node<E> x = first; x != null; x = x.next) {
  8. if (x.item == null) {
  9. //调用unlink方法删除指定节点
  10. unlink(x);
  11. return true;
  12. }
  13. }
  14. } else {
  15. for (Node<E> x = first; x != null; x = x.next) {
  16. if (o.equals(x.item)) {
  17. unlink(x);
  18. return true;
  19. }
  20. }
  21. }
  22. return false;
  23. }

  24. //删除指定位置的节点,其实和上面的方法差不多
  25. //通过node方法获得指定位置的节点,再通过unlink方法删除
  26. public E remove(int index) {
  27. checkElementIndex(index);
  28.  
  29. return unlink(node(index));
  30. }

  31. //删除指定节点
  32. E unlink(Node<E> x) {
  33. //获取x节点的元素,以及它上一个节点,和下一个节点
  34. final E element = x.item;
  35. final Node<E> next = x.next;
  36. final Node<E> prev = x.prev;
  37. //如果x的上一个节点为null,说明是首节点,将x的下一个节点设置为新的首节点
  38. //否则将x的上一节点设置为next,将x的上一节点设为null
  39. if (prev == null) {
  40. first = next;
  41. } else {
  42. prev.next = next;
  43. x.prev = null;
  44. }
  45. //如果x的下一节点为null,说明是尾节点,将x的上一节点设置新的尾节点
  46. //否则将x的上一节点设置x的上一节点,将x的下一节点设为null
  47. if (next == null) {
  48. last = prev;
  49. } else {
  50. next.prev = prev;
  51. x.next = null;
  52. }
  53. //将x节点的元素值设为null,等待垃圾收集器收集
  54. x.item = null;
  55. //链表节点个数减1
  56. size--;
  57. //将集合修改次数加1
  58. modCount++;
  59. //返回删除节点的元素值
  60. return element;
  61. }
删除首节点-removeFirst()
  1. //删除首节点
  2. public E remove() {
  3. return removeFirst();
  4. }
  5. //删除首节点
  6. public E removeFirst() {
  7. final Node<E> f = first;
  8. //如果首节点为null,说明是空链表,抛出异常
  9. if (f == null)
  10. throw new NoSuchElementException();
  11. return unlinkFirst(f);
  12. }
  13. //删除首节点
  14. private E unlinkFirst(Node<E> f) {
  15. //首节点的元素值
  16. final E element = f.item;
  17. //首节点的下一节点
  18. final Node<E> next = f.next;
  19. //将首节点的元素值和下一节点设为null,等待垃圾收集器收集
  20. f.item = null;
  21. f.next = null; // help GC
  22. //将next设置为新的首节点
  23. first = next;
  24. //如果next为null,说明说明链表中只有一个节点,把last也设为null
  25. //否则把next的上一节点设为null
  26. if (next == null)
  27. last = null;
  28. else
  29. next.prev = null;
  30. //链表节点个数减1
  31. size--;
  32. //将集合修改次数加1
  33. modCount++;
  34. //返回删除节点的元素值
  35. return element;
  36. }
删除尾节点-removeLast()
  1.  
  1. //删除尾节点
  2. public E removeLast() {
  3. final Node<E> l = last;
  4. //如果首节点为null,说明是空链表,抛出异常
  5. if (l == null)
  6. throw new NoSuchElementException();
  7. return unlinkLast(l);
  8. }
  9. private E unlinkLast(Node<E> l) {
  10. //尾节点的元素值
  11. final E element = l.item;
  12. //尾节点的上一节点
  13. final Node<E> prev = l.prev;
  14. //将尾节点的元素值和上一节点设为null,等待垃圾收集器收集
  15. l.item = null;
  16. l.prev = null; // help GC
  17. //将prev设置新的尾节点
  18. last = prev;
  19. //如果prev为null,说明说明链表中只有一个节点,把first也设为null
  20. //否则把prev的下一节点设为null
  21. if (prev == null)
  22. first = null;
  23. else
  24. prev.next = null;
  25. //链表节点个数减1
  26. size--;
  27. //将集合修改次数加1
  28. modCount++;
  29. //返回删除节点的元素值
  30. return element;
  31. }

其他方法也是类似的,比如查询方法 LinkedList提供了getgetFirstgetLast等方法获取节点元素值。

modCount属性的作用?

modCount属性代表为结构性修改( 改变list的size大小、以其他方式改变他导致正在进行迭代时出现错误的结果)的次数,该属性被Iterator以及ListIterator的实现类所使用,且很多非线程安全使用modCount属性。

初始化迭代器时会给这个modCount赋值,如果在遍历的过程中,一旦发现这个对象的modCount和迭代器存储的modCount不一样,Iterator或者ListIterator 将抛出ConcurrentModificationException异常,

这是jdk在面对迭代遍历的时候为了避免不确定性而采取的 fail-fast(快速失败)原则:

在线程不安全的集合中,如果使用迭代器的过程中,发现集合被修改,会抛出ConcurrentModificationExceptions错误,这就是fail-fast机制。对集合进行结构性修改时,modCount都会增加,在初始化迭代器时,modCount的值会赋给expectedModCount,在迭代的过程中,只要modCount改变了,int expectedModCount = modCount等式就不成立了,迭代器检测到这一点,就会抛出错误:urrentModificationExceptions


总结

ArrayList和LinkedList的区别、优缺点以及应用场景

区别:

  • ArrayList是实现了基于动态数组的数据结构,LinkedList是基于链表结构。

  • 对于随机访问的getset方法查询元素,ArrayList要优于LinkedList,因为LinkedList循环链表寻找元素。

  • 对于新增和删除操作addremoveLinkedList比较高效,因为ArrayList要移动数据。

优缺点:

  • ArrayListLinkedList而言,在末尾增加一个元素所花的开销都是固定的。对ArrayList而言,主要是在内部数组中增加一项,指向所添加的元素,偶尔可能会导致对数组重新进行分配;而对LinkedList而言,这个开销是 统一的,分配一个内部Entry对象。

  • ArrayList集合中添加或者删除一个元素时,当前的列表移动元素后面所有的元素都会被移动。而LinkedList集合中添加或者删除一个元素的开销是固定的。

  • LinkedList集合不支持 高效的随机随机访问(RandomAccess),因为可能产生二次项的行为。

  • ArrayList的空间浪费主要体现在在list列表的结尾预留一定的容量空间,而LinkedList的空间花费则体现在它的每一个元素都需要消耗相当的空间

应用场景:

ArrayList使用在查询比较多,但是插入和删除比较少的情况,而LinkedList用在查询比较少而插入删除比较多的情况

各位看官还可以吗?喜欢的话,动动手指点个,点个关注呗!!谢谢支持!

欢迎扫码关注,原创技术文章第一时间推出

面试:在面试中关于List(ArrayList、LinkedList)集合会怎么问呢?你该如何回答呢?的更多相关文章

  1. 在C语言中模仿java的LinkedList集合的使用(不要错过哦)

    在C语言中,多个数据的储存通常会用到数组.但是C语言的数组有个缺陷,就是固定长度,超过数组的最大长度就会溢出.怎样实现N个数储存起来而不被溢出呢. 学过java的都知道,java.util包里有一个L ...

  2. List ArrayList LinkedList 集合三

    因为List是有顺序的说以API中定义了 方法来处理List Collections 接口:用于操作List集合有排序sort(list);查找int binarySearch(List,obj);c ...

  3. 面试、笔试中常用的SQL语句(数据库知识必杀)一共50个!!!

    Student(S#,Sname,Sage,Ssex) 学生表  Course(C#,Cname,T#) 课程表  SC(S#,C#,score) 成绩表  Teacher(T#,Tname) 教师表 ...

  4. 前端面试之JavaScript中数组的方法!【残缺版!!】

    前端面试之JavaScript中数组常用的方法 7 join Array.join()方法将数组中所有元素都转化为字符串并连接在-起,返回最后生成的字 符串.可以指定一个可选的字符串在生成的字符串中来 ...

  5. 前端面试之JavaScript中的闭包!

    前端面试之JavaScript中的闭包! 闭包 闭包( closure )指有权访问另一个函数作用域中变量的函数. ----- JavaScript 高级程序设计 闭包其实可以理解为是一个函数 简单理 ...

  6. 前端面试之ES6中的继承!

    前端面试之ES6中的继承! ES6之前并没有给我们提供 extends继承.我们可以通过构造函数+原型对象模拟实现继承,被称为组合继承. 1 call() 两个作用: 1 调用这个函数! 2 修改函数 ...

  7. ArrayList, LinkedList, Vector - dudu:史上最详解

    ArrayList, LinkedList, Vector - dudu:史上最详解 我们来比较一下ArrayList, LinkedLIst和Vector它们之间的区别.BZ的JDK版本是1.7.0 ...

  8. ArrayList,LinkedList,Vector集合的认识

    最近在温习Java集合部分,花了三天时间读完了ArrayList与LinkedList以及Vector部分的源码.之前都是停留在简单使用ArrayList的API,读完源码看完不少文章后总算是对原理方 ...

  9. Java中Array与ArrayList的10个区别

    Array和ArrayList都是Java中两个重要的数据结构,在Java程序中经常使用.并且ArrayList在内部由Array支持,了解Java中的Array和ArrayList之间的差异对于成为 ...

随机推荐

  1. SpringCloud异常处理统一封装我来做-使用篇

    SpringCloud异常处理统一封装我来做-使用篇 简介 重复功能我来写.在 SpringBoot 项目里都有全局异常处理以及返回包装等,返回前端是带上succ.code.msg.data等字段.单 ...

  2. 蓝桥杯 试题 历届试题 对局匹配 DP解决

    问题描述 小明喜欢在一个围棋网站上找别人在线对弈.这个网站上所有注册用户都有一个积分,代表他的围棋水平. 小明发现网站的自动对局系统在匹配对手时,只会将积分差恰好是K的两名用户匹配在一起.如果两人分差 ...

  3. 07.django日志配置

    https://docs.djangoproject.com/en/3.0/topics/logging/ https://yiyibooks.cn/xx/python_352/library/log ...

  4. ShoneSharp语言(S#)的设计和使用介绍系列(11)—“类”披炫服靓妆化成“表”

    ShoneSharp语言(S#)的设计和使用介绍 系列(11)—“类”披炫服靓妆化成“表” 作者:Shone 声明:原创文章欢迎转载,但请注明出处,https://www.cnblogs.com/Sh ...

  5. xshell使用技巧

    XShell是一款Windows下的一款远程连接Linux主机的工具,类似的软件还有SecureCRT,putty等,但是个人感觉XShell好用,功能强大.. 一.复制和粘贴 linux的Shell ...

  6. ECharts使用教程

    引入 ECharts ECharts 3 开始不再强制使用 AMD 的方式按需引入,代码里也不再内置 AMD 加载器.因此引入方式简单了很多,只需要像普通的 JavaScript 库一样用 scrip ...

  7. PHP文件目录操作

    目录操作 is_dir ( $path ) 判断当前路径是否为目录 ,返回布尔 opendir ( $path ) 打开路径目录,返回资源 readdir ( $handle ) 读取当前打开目录下一 ...

  8. Docker安装常见的应用与将本地镜像推送到阿里云

    一.Docker安装常用的应用 1,docker安装mysql #拉取镜像mysql5.7 docker pull mysql:5.7 #启动容器(绑定对应的配置文件和日志,默认密码为123456) ...

  9. 【真相揭秘】requests获取网页编码乱码本质

    有没有被网页编码抓狂,怎么转都是乱码. 通过查看requests源代码,才发现是库本身历史原因造成的. 作者是严格http协议标准写这个库的,<HTTP权威指南>里第16章国际化里提到,如 ...

  10. CentOS 安装 git2.x.x 版本

    方法一 源码方式安装 第一步:卸载旧的git版本. $ yum remove git 第二步:下载git $ wget --no-check-certificate https://www.kerne ...