“This is the third day of my participation in the Gwen Challenge in November. Check out the details: The last Gwen Challenge in 2021.”

List is one of the important data structures in Java. We often come into contact with ArrayList, Vector and LinkedList, which all inherit from java.util.Collection interface. Let’s focus on ArrayList, as shown in the following class diagram

 

First, basic implementation

ArrayList: An array implementation encapsulates the advantages of operations on internal arrays: efficient operation reads, array-based implementations, null values, repeatable elements, ordered, asynchronous. Disadvantages: Since it is implemented by dynamic arrays, it is not suitable for frequent insert and delete operations on elements, as each insert and delete requires moving elements in the array. LinkedList: uses bidirectional LinkedList data structure advantages: LinkedList is implemented by double LinkedList, adding and deleting data of the underlying array is not required to move, the underlying array is implemented by LinkedList, only need to modify the LinkedList node pointer, high efficiency of element insertion and deletion. Disadvantages: Low traversal efficiency. Hashmaps are also related to double-linked lists.Copy the code

Second, the ArrayList

Implementation principle:

The underlying ArrayList is a variable-length array, basically the same as a Vector, except that Vector synchronizes the writeObjec()t and readObject() methods.

A static variable

Private static final int DEFAULT_CAPACITY = 10; private static final int DEFAULT_CAPACITY = 10; /** * Shared empty array instances for empty instances. */ private static final Object[] EMPTY_ELEMENTDATA = {}; /** Shared empty array instances are used for empty instances of the default size. */ private static final Object[] DEFAULTCAPACITY_EMPTY_ELEMENTDATA = {}; /** The array buffer that stores elements in the arraylist. */ transient Object[] elementData; // Non-private to simplify class access /** Arraylist size */ private int size;Copy the code

The add (E, E) method

Public Boolean add(E E) {// Check whether ensureCapacityInternal(size + 1); // insert the element to the last bit elementData[size++] = e; return true; } private void ensureCapacityInternal(int minCapacity) { ensureExplicitCapacity(calculateCapacity(elementData, minCapacity)); } private static int calculateCapacity(Object[] elementData, Int minCapacity) {// If DEFAULTCAPACITY_EMPTY_ELEMENTDATA is empty, 10 if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {return math. Max (DEFAULT_CAPACITY, minCapacity); } return minCapacity; } private void ensureExplicitCapacity(int minCapacity) { modCount++; If (minCapacity -elementdata. length > 0) // Expand grow(minCapacity); } private void grow(int minCapacity) { int oldCapacity = elementData.length; // the newCapacity is 1.5 times the oldCapacity int newCapacity = oldCapacity + (oldCapacity >> 1); If (newCapacity - minCapacity < 0) newCapacity = minCapacity; // if the newCapacity is smaller than the required capacity, the required capacity prevails. If (newCapacity -max_array_size > 0) newCapacity = hugeCapacity(minCapacity); if (newCapacity -max_array_size > 0) ElementData = array. copyOf(elementData, newCapacity); }Copy the code

The specific steps are as follows:

1. Check whether the system needs to be expanded.

2. If elementData is equal to DEFAULTCAPACITY_EMPTY_ELEMENTDATA, initialize the capacity to DEFAULT_CAPACITY.

3. The new capacity is 1.5 times of the oldCapacity (oldCapacity + (oldCapacity >> 1)). If the capacity is smaller than the required capacity, the required capacity prevails.

4. Create a new array and copy the old array to the new array.

Add (int index, E Element) method

Adds an element to the specified location.

Public void add(int index, E element) {// Check if rangeCheckForAdd(index); // Check whether ensureCapacityInternal(size + 1) needs to be expanded; Arraycopy (elementData, index, elementData, index + 1, size-index); arrayCopy (elementData, index, elementData, index + 1, size-index); ElementData [index] = element; // the size is increased by 1; } private void rangeCheckForAdd(int index) { if (index > size || index < 0) throw new IndexOutOfBoundsException(outOfBoundsMsg(index)); }Copy the code

The specific steps are as follows:

1. Check whether the index is out of bounds.

2. Check whether the system needs to be expanded.

3. Move all the elements inserted in the index one bit back.

4. Place the inserted element in the insert index position. 5. Size + 1;

AddAll method

Find the union of two sets.

Public Boolean addAll(Collection<? Extends E> c) {// Convert collection c to array Object[] a = c.toarray (); int numNew = a.length; // Check whether ensureCapacityInternal(size + numNew) needs to be expanded; System. arrayCopy (a, 0, elementData, size, numNew); // increase the size of c size += numNew; // return true if c is not empty, false return numNew! = 0; }Copy the code

Get (int index) method

Gets the element at the specified index position,

Public E get(int index) {// Check if rangeCheck(index); Return elementData(index); } private void rangeCheck(int index) { if (index >= size) throw new IndexOutOfBoundsException(outOfBoundsMsg(index)); } E elementData(int index) { return (E) elementData[index]; }Copy the code

The specific steps are as follows:

1. Check whether the index of crossing the line, here only to check whether the upper bound, if the upper throw IndexOutOfBoundsException abnormalities, if the lower bound is an ArrayIndexOutOfBoundsException anomalies.

2. Return the element at the index position;

Remove (int index) method

Deletes the element at the specified index position.

Public E remove(int index) {// Check if rangeCheck(index); modCount++; E oldValue = elementData(index); Int numMoved = size-index-1; int numMoved = size-index-1; if (numMoved > 0) System.arraycopy(elementData, index+1, elementData, index, numMoved); // Remove the last element to help GC elementData[--size] = null; // clear to let GC do its work return oldValue; }Copy the code

1. Check whether the index is out of bounds.

2. Get the element at the specified index position.

3. If the deleted bit is not the last one, the other elements move forward one bit.

4. Set the last position to NULL for GC collection.

5. Return the deleted element.

As you can see, the ArrayList deletes elements without shrinking.

conclusion

1. The ArrayList uses arrays to store elements internally. If the size of the array is insufficient, the ArrayList will be expanded.

2. ArrayList supports random access. Accessing elements through indexes is extremely fast and the time complexity is O(1);

3. ArrayList adds elements to tail very quickly, with an average time complexity of O(1);

4. ArrayList is slow to add elements to the middle because the average time to move elements is O(n);

5. ArrayList removes elements from the tail very quickly, with O(1) time complexity;

6. ArrayList is slow to remove elements from the middle because the average time complexity of moving elements is O(n);