diff --git a/35/target/ArrayListDemo.class b/35/target/ArrayListDemo.class new file mode 100644 index 0000000..98f8536 Binary files /dev/null and b/35/target/ArrayListDemo.class differ diff --git a/35/target/HashMapDemo$1.class b/35/target/HashMapDemo$1.class new file mode 100644 index 0000000..0ead949 Binary files /dev/null and b/35/target/HashMapDemo$1.class differ diff --git a/35/target/HashMapDemo.class b/35/target/HashMapDemo.class new file mode 100644 index 0000000..f97b55e Binary files /dev/null and b/35/target/HashMapDemo.class differ diff --git a/35/target/LinkedListDemo.class b/35/target/LinkedListDemo.class new file mode 100644 index 0000000..e2aa6f6 Binary files /dev/null and b/35/target/LinkedListDemo.class differ diff --git "a/week_01/04/ArrayList \346\272\220\347\240\201\345\210\206\346\236\220.md" "b/week_01/04/ArrayList \346\272\220\347\240\201\345\210\206\346\236\220.md" new file mode 100644 index 0000000..7ecd5c9 --- /dev/null +++ "b/week_01/04/ArrayList \346\272\220\347\240\201\345\210\206\346\236\220.md" @@ -0,0 +1,447 @@ +# ArrayList 源码分析 + + + +## 继承体系 + +- AbstractList +- List +- RandomAccess +- Cloneable +- java.io.Serializable + +> List 接口,是一种有顺序的集合,sequence (序列,顺序) +> +> index +> +> typically Duplicate elements (Unlike sets) + + + +> ArrayList Resiable-array, permits alll elements including null. +> +> roughly equivalent to Vector (synchronized) +> +> size,isEmpty,get,set,iterator,listIterator -> constant time +> +> add -> amortized(均摊) constant time +> +> other operations -> linear time (roughly) +> +> capacity >= list.size() ; capacity group automatically + + + +## 构造方法 + +1. 默认空构造 + + ​ + + ```java + /** + * Constructs an empty list with an initial capacity of ten. + * private static final Object[] DEFAULTCAPACITY_EMPTY_ELEMENTDATA = {}; + * 为什么说是10呢,可能是因为添加第一个元素时会扩展成10 + * Any empty ArrayList with elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA + * will be expanded to DEFAULT_CAPACITY when the first element is added + */ + public ArrayList() { + this.elementData = DEFAULTCAPACITY_EMPTY_ELEMENTDATA; + } + ``` + +2. 传入 initialCapacity的 构造 + + ```java + /** + * Constructs an empty list with the specified initial capacity. + * 当传入 初始容量为0的构造时,为什么和上面的空构造不用同一个 DEFAULTCAPACITY_EMPTY_ELEMENTDATA + * 这俩是为了什么设计而存在的呢? + * 在 DEFAULTCAPACITY_EMPTY_ELEMENTDATA 上的注释写了 + * ----------------------------- + * Shared empty array instance used for default sized empty instances. We + * distinguish this from EMPTY_ELEMENTDATA to know how much to inflate when + * first element is added. + * ----------------------------- + * 为什么存在呢? + * 大致意思就是如果你构造函数里传入了0,证明你这个故意为之,就是为了减少空间的使用, + * 默认构造呢 就是会使用默认的10容量 + * @param initialCapacity the initial capacity of the list + * @throws IllegalArgumentException if the specified initial capacity + * is negative + */ + public ArrayList(int initialCapacity) { + if (initialCapacity > 0) { + this.elementData = new Object[initialCapacity]; + } else if (initialCapacity == 0) { + this.elementData = EMPTY_ELEMENTDATA; + } else { + throw new IllegalArgumentException("Illegal Capacity: "+ + initialCapacity); + } + } + ``` + +3. 传入集合的构造 + + ```java + /** + * Constructs a list containing the elements of the specified + * collection, in the order they are returned by the collection's + * iterator. + * + * @param c the collection whose elements are to be placed into this list + * @throws NullPointerException if the specified collection is null + */ + public ArrayList(Collection c) { + elementData = c.toArray(); + if ((size = elementData.length) != 0) { + // defend against c.toArray (incorrectly) not returning Object[] + // (see e.g. https://bugs.openjdk.java.net/browse/JDK-6260652) + if (elementData.getClass() != Object[].class) + elementData = Arrays.copyOf(elementData, size, Object[].class); + } else { + // replace with empty array. + this.elementData = EMPTY_ELEMENTDATA; + } + } + ``` + + ```java + @HotSpotIntrinsicCandidate + public static T[] copyOf(U[] original, int newLength, Class newType) { + @SuppressWarnings("unchecked") + // 相同类型的元素 + T[] copy = ((Object)newType == (Object)Object[].class) + ? (T[]) new Object[newLength] + : (T[]) Array.newInstance(newType.getComponentType(), newLength); + System.arraycopy(original, 0, copy, 0, + Math.min(original.length, newLength)); + return copy; + } + ``` + + ## + +## Constant Time Methods + +1. size + + ```java + /** + * Returns the number of elements in this list. + * 返回list中的元素数量 + * @return the number of elements in this list + */ + public int size() { + return size; + } + ``` + +2. isEmpty + + ```java + /** + * Returns {@code true} if this list contains no elements. + * 如果list中不包含元素返回 true + * @return {@code true} if this list contains no elements + */ + public boolean isEmpty() { + return size == 0; + } + ``` + +3. get + + ```java + /** + * Returns the element at the specified position in this list. + * 返回list在指定位置的元素 + * @param index index of the element to return + * @return the element at the specified position in this list + * @throws IndexOutOfBoundsException {@inheritDoc} + */ + public E get(int index) { + Objects.checkIndex(index, size); + return elementData(index); + } + ``` + +4. set + + ```java + /** + * Replaces the element at the specified position in this list with + * the specified element. + * 替换指定位置元素 + * @param index index of the element to replace + * @param element element to be stored at the specified position + * @return the element previously at the specified position + * @throws IndexOutOfBoundsException {@inheritDoc} + */ + public E set(int index, E element) { + Objects.checkIndex(index, size); + E oldValue = elementData(index); + elementData[index] = element; + return oldValue; + } + ``` + +5. iterator + + ```java + /** + * Returns an iterator over the elements in this list in proper sequence. + * 返回此list的遍历器(以正确的顺序) + *

The returned iterator is fail-fast. + * + * @return an iterator over the elements in this list in proper sequence + */ + public Iterator iterator() { + // 具体实现后续再看 + return new Itr(); + } + ``` + +6. listIterator + + ```java + /** + * Returns a list iterator over the elements in this list (in proper + * sequence), starting at the specified position in the list. + * The specified index indicates the first element that would be + * returned by an initial call to {@link ListIterator#next next}. + * An initial call to {@link ListIterator#previous previous} would + * return the element with the specified index minus one. + * + *

The returned list iterator is fail-fast. + * + * @throws IndexOutOfBoundsException {@inheritDoc} + */ + public ListIterator listIterator(int index) { + rangeCheckForAdd(index); + return new ListItr(index); + } + ``` + + + +## 主要看 + +1. add + + ```java + /** + * Appends the specified element to the end of this list. + * 在列表中结尾增加指定元素 + * @param e element to be appended to this list + * @return {@code true} (as specified by {@link Collection#add}) + */ + public boolean add(E e) { + // 修改次数 + 1 + modCount++; + add(e, elementData, size); + return true; + } + + /** + * 此方法使方法字节码小于35,使得在C1编译循环中区分开来 + * This helper method split out from add(E) to keep method + * bytecode size under 35 (the -XX:MaxInlineSize default value), + * which helps when add(E) is called in a C1-compiled loop. + */ + private void add(E e, Object[] elementData, int s) { + if (s == elementData.length) + // 由于元素已满,所以调用grow函数 + elementData = grow(); + elementData[s] = e; + size = s + 1; + } + + + private Object[] grow() { + return grow(size + 1); + } + /** + * 增加容量确保集合可以存储传入的 minCapacity 数量 + * Increases the capacity to ensure that it can hold at least the + * number of elements specified by the minimum capacity argument. + * + * @param minCapacity the desired minimum capacity + * @throws OutOfMemoryError if minCapacity is less than zero + */ + private Object[] grow(int minCapacity) { + return elementData = Arrays.copyOf(elementData, + newCapacity(minCapacity)); + } + + + /** + * 返回 至少大于给定的 minCapacity大小,如果 1.5倍能够满足就返回。 + * 除非给定MAX_ARRAY_SIZE不然不会返回这个值 + * Returns a capacity at least as large as the given minimum capacity. + * Returns the current capacity increased by 50% if that suffices. + * Will not return a capacity greater than MAX_ARRAY_SIZE unless + * the given minimum capacity is greater than MAX_ARRAY_SIZE. + * + * @param minCapacity the desired minimum capacity + * @throws OutOfMemoryError if minCapacity is less than zero + */ + private int newCapacity(int minCapacity) { + // overflow-conscious code + int oldCapacity = elementData.length; + // 1.5倍 + int newCapacity = oldCapacity + (oldCapacity >> 1); + if (newCapacity - minCapacity <= 0) { + if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) + return Math.max(DEFAULT_CAPACITY, minCapacity); + if (minCapacity < 0) // overflow + // 溢出考虑 + throw new OutOfMemoryError(); + return minCapacity; + } + return (newCapacity - MAX_ARRAY_SIZE <= 0) + ? newCapacity + : hugeCapacity(minCapacity); + } + + private static int hugeCapacity(int minCapacity) { + if (minCapacity < 0) // overflow + throw new OutOfMemoryError(); + return (minCapacity > MAX_ARRAY_SIZE) + ? Integer.MAX_VALUE + : MAX_ARRAY_SIZE; + } + + /** + * 将元素插入list中指定位置。交换此时指定位置的现有元素,后移后置元素 + * Inserts the specified element at the specified position in this + * list. Shifts the element currently at that position (if any) and + * any subsequent elements to the right (adds one to their indices). + * + * @param index index at which the specified element is to be inserted + * @param element element to be inserted + * @throws IndexOutOfBoundsException {@inheritDoc} + */ + public void add(int index, E element) { + // 检查是否越界 + rangeCheckForAdd(index); + // 修改次数+1 + modCount++; + final int s; + Object[] elementData; + if ((s = size) == (elementData = this.elementData).length) + elementData = grow(); + // 调用系统数组拷贝 + System.arraycopy(elementData, index, + elementData, index + 1, + s - index); + elementData[index] = element; + size = s + 1; + } + + /** + * A version of rangeCheck used by add and addAll. + */ + private void rangeCheckForAdd(int index) { + if (index > size || index < 0) + throw new IndexOutOfBoundsException(outOfBoundsMsg(index)); + } + ``` + +2. addAll + + ```java + /** + * 将所有集合中元素加入此list的最后 + * Appends all of the elements in the specified collection to the end of + * this list, in the order that they are returned by the + * specified collection's Iterator. The behavior of this operation is + * undefined if the specified collection is modified while the operation + * is in progress. (This implies that the behavior of this call is + * undefined if the specified collection is this list, and this + * list is nonempty.) + * + * @param c collection containing elements to be added to this list + * @return {@code true} if this list changed as a result of the call + * @throws NullPointerException if the specified collection is null + */ + public boolean addAll(Collection c) { + // 转成数组 + Object[] a = c.toArray(); + // 修改数量 + 1 + modCount++; + // 增加的数量 还有空指针的风险 + int numNew = a.length; + if (numNew == 0) + return false; + Object[] elementData; + // 增加的位置 + final int s; + if (numNew > (elementData = this.elementData).length - (s = size)) + elementData = grow(s + numNew); + // 将集合c中的元素拷贝到最后 + System.arraycopy(a, 0, elementData, s, numNew); + size = s + numNew; + return true; + } + ``` + +3. remove + + ```java + /** + * 移除list中第一个出现的元素(如果存在的话),如果list不包含元素,那这个就不会改变。更正式来说就是移除低索 + * 引的元素 满足Objects.equals(o, get(i))的元素存在。 + * Removes the first occurrence of the specified element from this list, + * if it is present. If the list does not contain the element, it is + * unchanged. More formally, removes the element with the lowest index + * {@code i} such that + * {@code Objects.equals(o, get(i))} + * (if such an element exists). Returns {@code true} if this list + * contained the specified element (or equivalently, if this list + * changed as a result of the call). + * + * @param o element to be removed from this list, if present + * @return {@code true} if this list contained the specified element + */ + public boolean remove(Object o) { + final Object[] es = elementData; + final int size = this.size; + int i = 0; + // 代码块 + found: { + // 区分为空 + if (o == null) { + for (; i < size; i++) + if (es[i] == null) + break found; + // 不为空的 + } else { + for (; i < size; i++) + if (o.equals(es[i])) + break found; + } + return false; + } + fastRemove(es, i); + return true; + } + + /** + * 基础的移除方法 跳过界限检查不返回移除的元素 + * Private remove method that skips bounds checking and does not + * return the value removed. + */ + private void fastRemove(Object[] es, int i) { + modCount++; + final int newSize; + if ((newSize = size - 1) > i) + System.arraycopy(es, i + 1, es, i, newSize - i); + es[size = newSize] = null; + } + + ``` + + \ No newline at end of file diff --git "a/week_01/04/HashMap\346\272\220\347\240\201\345\210\206\346\236\220.md" "b/week_01/04/HashMap\346\272\220\347\240\201\345\210\206\346\236\220.md" new file mode 100644 index 0000000..048ec8e --- /dev/null +++ "b/week_01/04/HashMap\346\272\220\347\240\201\345\210\206\346\236\220.md" @@ -0,0 +1,1130 @@ +# HashMap + + + +## 继承体系 + +### 继承 + +​ AbstractMap + +### 实现 + +1. Map +2. Cloneable +3. Serializable + +## 主要属性 + +```java +/** + * 默认初始容量16 + * The default initial capacity - MUST be a power of two. + */ +static final int DEFAULT_INITIAL_CAPACITY = 1 << 4; // aka 16 + +/** + * 最大的容量 + * The maximum capacity, used if a higher value is implicitly specified + * by either of the constructors with arguments. + * MUST be a power of two <= 1<<30. + */ +static final int MAXIMUM_CAPACITY = 1 << 30; + +/** + * 默认加载因子 + * The load factor used when none specified in constructor. + */ +static final float DEFAULT_LOAD_FACTOR = 0.75f; + +/** + * 使用树的桶数量阈值 + * The bin count threshold for using a tree rather than list for a + * bin. Bins are converted to trees when adding an element to a + * bin with at least this many nodes. The value must be greater + * than 2 and should be at least 8 to mesh with assumptions in + * tree removal about conversion back to plain bins upon + * shrinkage. + */ +static final int TREEIFY_THRESHOLD = 8; + +/** + * 树退化成链表的阈值 + * The bin count threshold for untreeifying a (split) bin during a + * resize operation. Should be less than TREEIFY_THRESHOLD, and at + * most 6 to mesh with shrinkage detection under removal. + */ +static final int UNTREEIFY_THRESHOLD = 6; + +/** + * The smallest table capacity for which bins may be treeified. + * (Otherwise the table is resized if too many nodes in a bin.) + * Should be at least 4 * TREEIFY_THRESHOLD to avoid conflicts + * between resizing and treeification thresholds. + */ +static final int MIN_TREEIFY_CAPACITY = 64; +``` + + + +```java +/** + * 桶(表),第一次使用时才会初始化,必要时扩容 + * The table, initialized on first use, and resized as + * necessary. When allocated, length is always a power of two. + * (We also tolerate length zero in some operations to allow + * bootstrapping mechanics that are currently not needed.) + */ +transient Node[] table; + +/** + * 缓存键值对的entrySet + * Holds cached entrySet(). Note that AbstractMap fields are used + * for keySet() and values(). + */ +transient Set> entrySet; + +/** + * 键值对的数量 + * The number of key-value mappings contained in this map. + */ +transient int size; + +/** + * 修改次数 + * The number of times this HashMap has been structurally modified + * Structural modifications are those that change the number of mappings in + * the HashMap or otherwise modify its internal structure (e.g., + * rehash). This field is used to make iterators on Collection-views of + * the HashMap fail-fast. (See ConcurrentModificationException). + */ +transient int modCount; + +/** + * 下一个要重新resize的大小 + * The next size value at which to resize (capacity * load factor). + * + * @serial + */ +// (The javadoc description is true upon serialization. +// Additionally, if the table array has not been allocated, this +// field holds the initial array capacity, or zero signifying +// DEFAULT_INITIAL_CAPACITY.) +int threshold; + +/** + * 加载因子 + * The load factor for the hash table. + * + * @serial + */ +final float loadFactor; +``` + +## 构造函数 + +1. 空构造 + + ```java + /** + * 默认值 16 load factor .75 + * Constructs an empty HashMap with the default initial capacity + * (16) and the default load factor (0.75). + */ + public HashMap() { + this.loadFactor = DEFAULT_LOAD_FACTOR; // all other fields defaulted + } + ``` + +2. 传入Map的构造 + + ```java + /** + * 构造足够承载目标的容量 + * Constructs a new HashMap with the same mappings as the + * specified Map. The HashMap is created with + * default load factor (0.75) and an initial capacity sufficient to + * hold the mappings in the specified Map. + * + * @param m the map whose mappings are to be placed in this map + * @throws NullPointerException if the specified map is null + */ + public HashMap(Map m) { + this.loadFactor = DEFAULT_LOAD_FACTOR; + putMapEntries(m, false); + } + + /** + * Implements Map.putAll and Map constructor. + * + * @param m the map + * @param evict false when initially constructing this map, else + * true (relayed to method afterNodeInsertion). + */ + final void putMapEntries(Map m, boolean evict) { + int s = m.size(); + if (s > 0) { + if (table == null) { // pre-size + // 除以加载因子 + 1 防止因加入全部元素而要扩容影响性能 + float ft = ((float)s / loadFactor) + 1.0F; + // 防止大于 MAXIMUM_CAPACITY + int t = ((ft < (float)MAXIMUM_CAPACITY) ? + (int)ft : MAXIMUM_CAPACITY); + // 修改threshold值 + if (t > threshold) + threshold = tableSizeFor(t); + } + else if (s > threshold) + // 查看如下 + resize(); + for (Map.Entry e : m.entrySet()) { + K key = e.getKey(); + V value = e.getValue(); + // 进行存值操作 + putVal(hash(key), key, value, false, evict); + } + } + } + ``` + + ```java + /** + * 初始化或将table双倍。如果是初始化,通过初始化容量threshold来分配, + * 然而由于我们使用了二次幂的扩容方式,元素可能存储在原索引位置,或者移动到二次幂的新表中 + * Initializes or doubles table size. If null, allocates in + * accord with initial capacity target held in field threshold. + * Otherwise, because we are using power-of-two expansion, the + * elements from each bin must either stay at same index, or move + * with a power of two offset in the new table. + * + * @return the table + */ + final Node[] resize() { + // 旧的表 + Node[] oldTab = table; + // 过去的容量 + int oldCap = (oldTab == null) ? 0 : oldTab.length; + // 过去的扩容指标 + int oldThr = threshold; + int newCap, newThr = 0; + if (oldCap > 0) { + if (oldCap >= MAXIMUM_CAPACITY) { + threshold = Integer.MAX_VALUE; + return oldTab; + } + // 旧容量 >= DEFAULT_INITIAL_CAPACITY && 新容量 = 旧容量 * 2 < MAXIMUM_CAPACITY + else if ((newCap = oldCap << 1) < MAXIMUM_CAPACITY && + oldCap >= DEFAULT_INITIAL_CAPACITY) + // 这种就是翻倍的情况 + newThr = oldThr << 1; // double threshold + } + else if (oldThr > 0) // initial capacity was placed in threshold + newCap = oldThr; + else { // zero initial threshold signifies using defaults + // 初始默认值 + newCap = DEFAULT_INITIAL_CAPACITY; + newThr = (int)(DEFAULT_LOAD_FACTOR * DEFAULT_INITIAL_CAPACITY); + } + if (newThr == 0) { + float ft = (float)newCap * loadFactor; + newThr = (newCap < MAXIMUM_CAPACITY && ft < (float)MAXIMUM_CAPACITY ? + (int)ft : Integer.MAX_VALUE); + } + threshold = newThr; + @SuppressWarnings({"rawtypes","unchecked"}) + Node[] newTab = (Node[])new Node[newCap]; + table = newTab; + if (oldTab != null) { + for (int j = 0; j < oldCap; ++j) { + Node e; + if ((e = oldTab[j]) != null) { + // gc help + oldTab[j] = null; + if (e.next == null) + newTab[e.hash & (newCap - 1)] = e; + else if (e instanceof TreeNode) + ((TreeNode)e).split(this, newTab, j, oldCap); + else { // preserve order + Node loHead = null, loTail = null; + Node hiHead = null, hiTail = null; + Node next; + do { + next = e.next; + if ((e.hash & oldCap) == 0) { + if (loTail == null) + loHead = e; + else + loTail.next = e; + loTail = e; + } + else { + if (hiTail == null) + hiHead = e; + else + hiTail.next = e; + hiTail = e; + } + } while ((e = next) != null); + if (loTail != null) { + loTail.next = null; + newTab[j] = loHead; + } + if (hiTail != null) { + hiTail.next = null; + newTab[j + oldCap] = hiHead; + } + } + } + } + } + return newTab; + } + ``` + + ```java + /** + * 实现Map.put 和 相关方法 + * Implements Map.put and related methods. + * + * @param hash hash for key + * @param key the key + * @param value the value to put + * @param onlyIfAbsent if true, don't change existing value 如果为true 不需要改变已存在的值 + * @param evict if false, the table is in creation mode. 如果为false ,表是在创建模式 + * @return previous value, or null if none + */ + final V putVal(int hash, K key, V value, boolean onlyIfAbsent, + boolean evict) { + Node[] tab; Node p; int n, i; + // 容器没有初始化的情况下 需要调用resize + if ((tab = table) == null || (n = tab.length) == 0) + n = (tab = resize()).length; + // 如果该位置为空的情况直接赋值 即 桶的位置上为空 + if ((p = tab[i = (n - 1) & hash]) == null) + tab[i] = newNode(hash, key, value, null); + else { + // 此位置已存在桶 + Node e; K k; + if (p.hash == hash && + ((k = p.key) == key || (key != null && key.equals(k)))) + e = p; + else if (p instanceof TreeNode) + // 如果已经是树了,那么调用TreeNode#putTreeVal + e = ((TreeNode)p).putTreeVal(this, tab, hash, key, value); + else { + for (int binCount = 0; ; ++binCount) { + if ((e = p.next) == null) { + p.next = newNode(hash, key, value, null); + // 如果插入了新数据后链表长度大于8,那么就要进行树化 + if (binCount >= TREEIFY_THRESHOLD - 1) // -1 for 1st + treeifyBin(tab, hash); + break; + } + if (e.hash == hash && + ((k = e.key) == key || (key != null && key.equals(k)))) + break; + p = e; + } + } + // 找到相同key的元素 + if (e != null) { // existing mapping for key + V oldValue = e.value; + // 是否要替换旧值 + if (!onlyIfAbsent || oldValue == null) + e.value = value; + afterNodeAccess(e); + return oldValue; + } + } + ++modCount; + if (++size > threshold) + resize(); + afterNodeInsertion(evict); + return null; + } + ``` + + ```java + /** + * Replaces all linked nodes in bin at index for given hash unless + * table is too small, in which case resizes instead. + */ + final void treeifyBin(Node[] tab, int hash) { + int n, index; Node e; + if (tab == null || (n = tab.length) < MIN_TREEIFY_CAPACITY) + resize(); + else if ((e = tab[index = (n - 1) & hash]) != null) { + TreeNode hd = null, tl = null; + // 把所有节点转化为树节点 + do { + TreeNode p = replacementTreeNode(e, null); + if (tl == null) + hd = p; + else { + p.prev = tl; + tl.next = p; + } + tl = p; + } while ((e = e.next) != null); + if ((tab[index] = hd) != null) + hd.treeify(tab); + } + } + ``` + +3. + + + +## 内部类 + +### 简单看 + + + +### 仔细看 + +1. Node + + ```java + /** + * 基础hash桶节点 + * Basic hash bin node, used for most entries. (See below for + * TreeNode subclass, and in LinkedHashMap for its Entry subclass.) + */ + static class Node implements Map.Entry { + final int hash; + final K key; + V value; + // 单向链表 + Node next; + + Node(int hash, K key, V value, Node next) { + this.hash = hash; + this.key = key; + this.value = value; + this.next = next; + } + + public final K getKey() { return key; } + public final V getValue() { return value; } + public final String toString() { return key + "=" + value; } + + public final int hashCode() { + return Objects.hashCode(key) ^ Objects.hashCode(value); + } + + public final V setValue(V newValue) { + V oldValue = value; + value = newValue; + return oldValue; + } + + public final boolean equals(Object o) { + if (o == this) + return true; + if (o instanceof Map.Entry) { + Map.Entry e = (Map.Entry)o; + if (Objects.equals(key, e.getKey()) && + Objects.equals(value, e.getValue())) + return true; + } + return false; + } + } + ``` + + + +2. TreeNode + + ```java + /** + * 树桶,继承了 LinkedHashMap.Entry,所以即可以作为常规或链式节点 + * Entry for Tree bins. Extends LinkedHashMap.Entry (which in turn + * extends Node) so can be used as extension of either regular or + * linked node. + */ + static final class TreeNode extends LinkedHashMap.Entry { + // 父亲节点 + TreeNode parent; // red-black tree links + // 左节点 + TreeNode left; + // 右节点 + TreeNode right; + TreeNode prev; // needed to unlink next upon deletion + boolean red; + // 构造方法 + TreeNode(int hash, K key, V val, Node next) { + super(hash, key, val, next); + } + + /** + * Returns root of tree containing this node. + * 树的根 + */ + final TreeNode root() { + for (TreeNode r = this, p;;) { + if ((p = r.parent) == null) + return r; + r = p; + } + } + + /** + * 确保给的根是头结点,将Root移到最前面 + * Ensures that the given root is the first node of its bin. + */ + static void moveRootToFront(Node[] tab, TreeNode root) { + int n; + if (root != null && tab != null && (n = tab.length) > 0) { + int index = (n - 1) & root.hash; + TreeNode first = (TreeNode)tab[index]; + if (root != first) { + Node rn; + tab[index] = root; + TreeNode rp = root.prev; + if ((rn = root.next) != null) + ((TreeNode)rn).prev = rp; + if (rp != null) + rp.next = rn; + if (first != null) + first.prev = root; + root.next = first; + root.prev = null; + } + assert checkInvariants(root); + } + } + + /** + * 找到给定hash值和key的Treenode + * Finds the node starting at root p with the given hash and key. + * The kc argument caches comparableClassFor(key) upon first use + * comparing keys. + */ + final TreeNode find(int h, Object k, Class kc) { + TreeNode p = this; + do { + int ph, dir; K pk; + TreeNode pl = p.left, pr = p.right, q; + if ((ph = p.hash) > h) + p = pl; + else if (ph < h) + p = pr; + else if ((pk = p.key) == k || (k != null && k.equals(pk))) + return p; + else if (pl == null) + p = pr; + else if (pr == null) + p = pl; + else if ((kc != null || + (kc = comparableClassFor(k)) != null) && + (dir = compareComparables(kc, k, pk)) != 0) + p = (dir < 0) ? pl : pr; + else if ((q = pr.find(h, k, kc)) != null) + return q; + else + p = pl; + } while (p != null); + return null; + } + + /** + * Calls find for root node. + */ + final TreeNode getTreeNode(int h, Object k) { + return ((parent != null) ? root() : this).find(h, k, null); + } + + /** + * Tie-breaking utility for ordering insertions when equal + * hashCodes and non-comparable. We don't require a total + * order, just a consistent insertion rule to maintain + * equivalence across rebalancings. Tie-breaking further than + * necessary simplifies testing a bit. + */ + static int tieBreakOrder(Object a, Object b) { + int d; + if (a == null || b == null || + (d = a.getClass().getName(). + compareTo(b.getClass().getName())) == 0) + d = (System.identityHashCode(a) <= System.identityHashCode(b) ? + -1 : 1); + return d; + } + + /** + * Forms tree of the nodes linked from this node. + */ + final void treeify(Node[] tab) { + TreeNode root = null; + for (TreeNode x = this, next; x != null; x = next) { + next = (TreeNode)x.next; + x.left = x.right = null; + if (root == null) { + x.parent = null; + x.red = false; + root = x; + } + else { + K k = x.key; + int h = x.hash; + Class kc = null; + for (TreeNode p = root;;) { + int dir, ph; + K pk = p.key; + if ((ph = p.hash) > h) + dir = -1; + else if (ph < h) + dir = 1; + else if ((kc == null && + (kc = comparableClassFor(k)) == null) || + (dir = compareComparables(kc, k, pk)) == 0) + dir = tieBreakOrder(k, pk); + + TreeNode xp = p; + if ((p = (dir <= 0) ? p.left : p.right) == null) { + x.parent = xp; + if (dir <= 0) + xp.left = x; + else + xp.right = x; + root = balanceInsertion(root, x); + break; + } + } + } + } + moveRootToFront(tab, root); + } + + /** + * Returns a list of non-TreeNodes replacing those linked from + * this node. + */ + final Node untreeify(HashMap map) { + Node hd = null, tl = null; + for (Node q = this; q != null; q = q.next) { + Node p = map.replacementNode(q, null); + if (tl == null) + hd = p; + else + tl.next = p; + tl = p; + } + return hd; + } + + /** + * Tree version of putVal. + */ + final TreeNode putTreeVal(HashMap map, Node[] tab, + int h, K k, V v) { + Class kc = null; + boolean searched = false; + TreeNode root = (parent != null) ? root() : this; + for (TreeNode p = root;;) { + int dir, ph; K pk; + if ((ph = p.hash) > h) + dir = -1; + else if (ph < h) + dir = 1; + else if ((pk = p.key) == k || (k != null && k.equals(pk))) + return p; + else if ((kc == null && + (kc = comparableClassFor(k)) == null) || + (dir = compareComparables(kc, k, pk)) == 0) { + if (!searched) { + TreeNode q, ch; + searched = true; + if (((ch = p.left) != null && + (q = ch.find(h, k, kc)) != null) || + ((ch = p.right) != null && + (q = ch.find(h, k, kc)) != null)) + return q; + } + dir = tieBreakOrder(k, pk); + } + + TreeNode xp = p; + if ((p = (dir <= 0) ? p.left : p.right) == null) { + Node xpn = xp.next; + TreeNode x = map.newTreeNode(h, k, v, xpn); + if (dir <= 0) + xp.left = x; + else + xp.right = x; + xp.next = x; + x.parent = x.prev = xp; + if (xpn != null) + ((TreeNode)xpn).prev = x; + moveRootToFront(tab, balanceInsertion(root, x)); + return null; + } + } + } + + /** + * Removes the given node, that must be present before this call. + * This is messier than typical red-black deletion code because we + * cannot swap the contents of an interior node with a leaf + * successor that is pinned by "next" pointers that are accessible + * independently during traversal. So instead we swap the tree + * linkages. If the current tree appears to have too few nodes, + * the bin is converted back to a plain bin. (The test triggers + * somewhere between 2 and 6 nodes, depending on tree structure). + */ + final void removeTreeNode(HashMap map, Node[] tab, + boolean movable) { + int n; + if (tab == null || (n = tab.length) == 0) + return; + int index = (n - 1) & hash; + TreeNode first = (TreeNode)tab[index], root = first, rl; + TreeNode succ = (TreeNode)next, pred = prev; + if (pred == null) + tab[index] = first = succ; + else + pred.next = succ; + if (succ != null) + succ.prev = pred; + if (first == null) + return; + if (root.parent != null) + root = root.root(); + if (root == null + || (movable + && (root.right == null + || (rl = root.left) == null + || rl.left == null))) { + tab[index] = first.untreeify(map); // too small + return; + } + TreeNode p = this, pl = left, pr = right, replacement; + if (pl != null && pr != null) { + TreeNode s = pr, sl; + while ((sl = s.left) != null) // find successor + s = sl; + boolean c = s.red; s.red = p.red; p.red = c; // swap colors + TreeNode sr = s.right; + TreeNode pp = p.parent; + if (s == pr) { // p was s's direct parent + p.parent = s; + s.right = p; + } + else { + TreeNode sp = s.parent; + if ((p.parent = sp) != null) { + if (s == sp.left) + sp.left = p; + else + sp.right = p; + } + if ((s.right = pr) != null) + pr.parent = s; + } + p.left = null; + if ((p.right = sr) != null) + sr.parent = p; + if ((s.left = pl) != null) + pl.parent = s; + if ((s.parent = pp) == null) + root = s; + else if (p == pp.left) + pp.left = s; + else + pp.right = s; + if (sr != null) + replacement = sr; + else + replacement = p; + } + else if (pl != null) + replacement = pl; + else if (pr != null) + replacement = pr; + else + replacement = p; + if (replacement != p) { + TreeNode pp = replacement.parent = p.parent; + if (pp == null) + root = replacement; + else if (p == pp.left) + pp.left = replacement; + else + pp.right = replacement; + p.left = p.right = p.parent = null; + } + + TreeNode r = p.red ? root : balanceDeletion(root, replacement); + + if (replacement == p) { // detach + TreeNode pp = p.parent; + p.parent = null; + if (pp != null) { + if (p == pp.left) + pp.left = null; + else if (p == pp.right) + pp.right = null; + } + } + if (movable) + moveRootToFront(tab, r); + } + + /** + * Splits nodes in a tree bin into lower and upper tree bins, + * or untreeifies if now too small. Called only from resize; + * see above discussion about split bits and indices. + * + * @param map the map + * @param tab the table for recording bin heads + * @param index the index of the table being split + * @param bit the bit of hash to split on + */ + final void split(HashMap map, Node[] tab, int index, int bit) { + TreeNode b = this; + // Relink into lo and hi lists, preserving order + TreeNode loHead = null, loTail = null; + TreeNode hiHead = null, hiTail = null; + int lc = 0, hc = 0; + for (TreeNode e = b, next; e != null; e = next) { + next = (TreeNode)e.next; + e.next = null; + if ((e.hash & bit) == 0) { + if ((e.prev = loTail) == null) + loHead = e; + else + loTail.next = e; + loTail = e; + ++lc; + } + else { + if ((e.prev = hiTail) == null) + hiHead = e; + else + hiTail.next = e; + hiTail = e; + ++hc; + } + } + + if (loHead != null) { + if (lc <= UNTREEIFY_THRESHOLD) + tab[index] = loHead.untreeify(map); + else { + tab[index] = loHead; + if (hiHead != null) // (else is already treeified) + loHead.treeify(tab); + } + } + if (hiHead != null) { + if (hc <= UNTREEIFY_THRESHOLD) + tab[index + bit] = hiHead.untreeify(map); + else { + tab[index + bit] = hiHead; + if (loHead != null) + hiHead.treeify(tab); + } + } + } + + /* ------------------------------------------------------------ */ + // Red-black tree methods, all adapted from CLR + // 左旋 + static TreeNode rotateLeft(TreeNode root, + TreeNode p) { + TreeNode r, pp, rl; + if (p != null && (r = p.right) != null) { + if ((rl = p.right = r.left) != null) + rl.parent = p; + if ((pp = r.parent = p.parent) == null) + (root = r).red = false; + else if (pp.left == p) + pp.left = r; + else + pp.right = r; + r.left = p; + p.parent = r; + } + return root; + } + // 右旋 + static TreeNode rotateRight(TreeNode root, + TreeNode p) { + TreeNode l, pp, lr; + if (p != null && (l = p.left) != null) { + if ((lr = p.left = l.right) != null) + lr.parent = p; + if ((pp = l.parent = p.parent) == null) + (root = l).red = false; + else if (pp.right == p) + pp.right = l; + else + pp.left = l; + l.right = p; + p.parent = l; + } + return root; + } + // 增加平衡 + static TreeNode balanceInsertion(TreeNode root, + TreeNode x) { + x.red = true; + for (TreeNode xp, xpp, xppl, xppr;;) { + if ((xp = x.parent) == null) { + x.red = false; + return x; + } + else if (!xp.red || (xpp = xp.parent) == null) + return root; + if (xp == (xppl = xpp.left)) { + if ((xppr = xpp.right) != null && xppr.red) { + xppr.red = false; + xp.red = false; + xpp.red = true; + x = xpp; + } + else { + if (x == xp.right) { + root = rotateLeft(root, x = xp); + xpp = (xp = x.parent) == null ? null : xp.parent; + } + if (xp != null) { + xp.red = false; + if (xpp != null) { + xpp.red = true; + root = rotateRight(root, xpp); + } + } + } + } + else { + if (xppl != null && xppl.red) { + xppl.red = false; + xp.red = false; + xpp.red = true; + x = xpp; + } + else { + if (x == xp.left) { + root = rotateRight(root, x = xp); + xpp = (xp = x.parent) == null ? null : xp.parent; + } + if (xp != null) { + xp.red = false; + if (xpp != null) { + xpp.red = true; + root = rotateLeft(root, xpp); + } + } + } + } + } + } + // 删除平衡 + static TreeNode balanceDeletion(TreeNode root, + TreeNode x) { + for (TreeNode xp, xpl, xpr;;) { + if (x == null || x == root) + return root; + else if ((xp = x.parent) == null) { + x.red = false; + return x; + } + else if (x.red) { + x.red = false; + return root; + } + else if ((xpl = xp.left) == x) { + if ((xpr = xp.right) != null && xpr.red) { + xpr.red = false; + xp.red = true; + root = rotateLeft(root, xp); + xpr = (xp = x.parent) == null ? null : xp.right; + } + if (xpr == null) + x = xp; + else { + TreeNode sl = xpr.left, sr = xpr.right; + if ((sr == null || !sr.red) && + (sl == null || !sl.red)) { + xpr.red = true; + x = xp; + } + else { + if (sr == null || !sr.red) { + if (sl != null) + sl.red = false; + xpr.red = true; + root = rotateRight(root, xpr); + xpr = (xp = x.parent) == null ? + null : xp.right; + } + if (xpr != null) { + xpr.red = (xp == null) ? false : xp.red; + if ((sr = xpr.right) != null) + sr.red = false; + } + if (xp != null) { + xp.red = false; + root = rotateLeft(root, xp); + } + x = root; + } + } + } + else { // symmetric + if (xpl != null && xpl.red) { + xpl.red = false; + xp.red = true; + root = rotateRight(root, xp); + xpl = (xp = x.parent) == null ? null : xp.left; + } + if (xpl == null) + x = xp; + else { + TreeNode sl = xpl.left, sr = xpl.right; + if ((sl == null || !sl.red) && + (sr == null || !sr.red)) { + xpl.red = true; + x = xp; + } + else { + if (sl == null || !sl.red) { + if (sr != null) + sr.red = false; + xpl.red = true; + root = rotateLeft(root, xpl); + xpl = (xp = x.parent) == null ? + null : xp.left; + } + if (xpl != null) { + xpl.red = (xp == null) ? false : xp.red; + if ((sl = xpl.left) != null) + sl.red = false; + } + if (xp != null) { + xp.red = false; + root = rotateRight(root, xp); + } + x = root; + } + } + } + } + } + + /** + * Recursive invariant check + */ + static boolean checkInvariants(TreeNode t) { + TreeNode tp = t.parent, tl = t.left, tr = t.right, + tb = t.prev, tn = (TreeNode)t.next; + if (tb != null && tb.next != t) + return false; + if (tn != null && tn.prev != t) + return false; + if (tp != null && t != tp.left && t != tp.right) + return false; + if (tl != null && (tl.parent != t || tl.hash > t.hash)) + return false; + if (tr != null && (tr.parent != t || tr.hash < t.hash)) + return false; + if (t.red && tl != null && tl.red && tr != null && tr.red) + return false; + if (tl != null && !checkInvariants(tl)) + return false; + if (tr != null && !checkInvariants(tr)) + return false; + return true; + } + } + ``` + + + +## 主要方法 + +### 简单看 + +```java +/** + * 返回指定key映射的值,如果不存在此Key 就返回null + * Returns the value to which the specified key is mapped, + * or {@code null} if this map contains no mapping for the key. + * + *

More formally, if this map contains a mapping from a key + * {@code k} to a value {@code v} such that {@code (key==null ? k==null : + * key.equals(k))}, then this method returns {@code v}; otherwise + * it returns {@code null}. (There can be at most one such mapping.) + * + *

A return value of {@code null} does not necessarily + * indicate that the map contains no mapping for the key; it's also + * possible that the map explicitly maps the key to {@code null}. + * The {@link #containsKey containsKey} operation may be used to + * distinguish these two cases. + * + * @see #put(Object, Object) + */ +public V get(Object key) { + Node e; + return (e = getNode(hash(key), key)) == null ? null : e.value; +} + +/** + * Implements Map.get and related methods. + * + * @param hash hash for key + * @param key the key + * @return the node, or null if none + */ +final Node getNode(int hash, Object key) { + Node[] tab; Node first, e; int n; K k; + if ((tab = table) != null && (n = tab.length) > 0 && + (first = tab[(n - 1) & hash]) != null) { + // 为什么要check头节点 因为 map一开始并不做初始化? + if (first.hash == hash && // always check first node + ((k = first.key) == key || (key != null && key.equals(k)))) + return first; + if ((e = first.next) != null) { + // 第一个元素是树那么就按照树的方式查找 + if (first instanceof TreeNode) + return ((TreeNode)first).getTreeNode(hash, key); + // 否则遍历整个链表查询 + do { + if (e.hash == hash && + ((k = e.key) == key || (key != null && key.equals(k)))) + return e; + } while ((e = e.next) != null); + } + } + return null; +} +``` + + + +```java +/** + * 在这个map中将键值关联,如果map之前存在此key,老的值会被替代 + * Associates the specified value with the specified key in this map. + * If the map previously contained a mapping for the key, the old + * value is replaced. + * + * @param key key with which the specified value is to be associated + * @param value value to be associated with the specified key + * @return the previous value associated with key, or + * null if there was no mapping for key. + * (A null return can also indicate that the map + * previously associated null with key.) + */ +public V put(K key, V value) { + // 此方法上面有分析 + return putVal(hash(key), key, value, false, true); +} +``` + +### 仔细读 + diff --git "a/week_01/04/LinkedList\346\272\220\347\240\201\345\210\206\346\236\220.md" "b/week_01/04/LinkedList\346\272\220\347\240\201\345\210\206\346\236\220.md" new file mode 100644 index 0000000..c0164af --- /dev/null +++ "b/week_01/04/LinkedList\346\272\220\347\240\201\345\210\206\346\236\220.md" @@ -0,0 +1,401 @@ +# LinkedList + +> ``` +> * Doubly-linked list implementation of the {@code List} and {@code Deque} +> * interfaces. Implements all optional list operations, and permits all +> * elements (including {@code null}). +> ``` +> +> 双向链表实现了List接口和Deque,实现了所有的list操作,允许包含空的所有元素。 +> +> ** not synchronized ** -> Collections#synchronizedList 方法调用 +> +> 实现了 List Deque + + + + + +## 主要属性 + +```java +// 元素的个数 +transient int size = 0; + +/** + * 指向第一个节点 + * Pointer to first node. + * 什么意思? + * Invariant: (first == null && last == null) || + * (first.prev == null && first.item != null) + */ +transient Node first; + +/** + * 指向最后一个节点 + * Pointer to last node. + * Invariant: (first == null && last == null) || + * (last.next == null && last.item != null) + */ +transient Node last; +``` + + + +## 构造方法 + +1. LinkedList() + + ```java + /** + * 构造空的集合 + * Constructs an empty list. + */ + public LinkedList() { + } + ``` + +2. LinkedList(Collection c) + + ```java + /** + * 构造一个list集合包含传入的集合的所有元素,他们的顺序按照 集合c 所iterator的返回顺序 + * Constructs a list containing the elements of the specified + * collection, in the order they are returned by the collection's + * iterator. + * + * @param c the collection whose elements are to be placed into this list + * @throws NullPointerException if the specified collection is null + */ + public LinkedList(Collection c) { + this(); + // 将集合加到此list中 后续再看 + addAll(c); + } + ``` + + + + + + + +## 方法 + +### 简单看 + +1. 查 + + ```java + /** + * 返回第一个元素 O1 + * Returns the first element in this list. + * + * @return the first element in this list + * @throws NoSuchElementException if this list is empty + */ + public E getFirst() { + final Node f = first; + if (f == null) + throw new NoSuchElementException(); + return f.item; + } + + /** + * 返回最后一个元素 O1 + * Returns the last element in this list. + * + * @return the last element in this list + * @throws NoSuchElementException if this list is empty + */ + public E getLast() { + final Node l = last; + if (l == null) + throw new NoSuchElementException(); + return l.item; + } + + ``` + +2. 删 + + ```java + /** + * 移出并返回第一元素 + * Removes and returns the first element from this list. + * + * @return the first element from this list + * @throws NoSuchElementException if this list is empty + */ + public E removeFirst() { + final Node f = first; + if (f == null) + throw new NoSuchElementException(); + return unlinkFirst(f); + } + + /** + * 移出并返回最后一个元素 + * Removes and returns the last element from this list. + * + * @return the last element from this list + * @throws NoSuchElementException if this list is empty + */ + public E removeLast() { + final Node l = last; + if (l == null) + throw new NoSuchElementException(); + return unlinkLast(l); + } + ``` + + + +3. + + + +### 细看 + +1. addAll + + ```java + /** + * 将传入的集合c全部加入到此集合list中,他们的顺序是按照collection c iterator的顺序。 + * 当传入的collection c 在操作在进行中时,被修改了。此操作的表现 undefined + * Appends all of the elements in the specified collection to the end of + * this list, in the order that they are returned by the specified + * collection's iterator. The behavior of this operation is undefined if + * the specified collection is modified while the operation is in + * progress. (Note that this will occur if the specified collection is + * this list, and it's nonempty.) + * + * @param c collection containing elements to be added to this list + * @return {@code true} if this list changed as a result of the call + * @throws NullPointerException if the specified collection is null + */ + public boolean addAll(Collection c) { + return addAll(size, c); + } + + /** + * 将传入的集合的所有元素加入到此list中(从指定位置开始),将当前位置交换,右边的往后移。 + * 新增的元素会以集合的 iterator的函数顺序 + * Inserts all of the elements in the specified collection into this + * list, starting at the specified position. Shifts the element + * currently at that position (if any) and any subsequent elements to + * the right (increases their indices). The new elements will appear + * in the list in the order that they are returned by the + * specified collection's iterator. + * + * @param index index at which to insert the first element + * from the specified collection + * @param c collection containing elements to be added to this list + * @return {@code true} if this list changed as a result of the call + * @throws IndexOutOfBoundsException {@inheritDoc} + * @throws NullPointerException if the specified collection is null + */ + public boolean addAll(int index, Collection c) { + // 检查索引是否越界 + checkPositionIndex(index); + + Object[] a = c.toArray(); + int numNew = a.length; + if (numNew == 0) + return false; + // pred 要加入的前一个,succ 要插入的元素的位置 + Node pred, succ; + if (index == size) { + succ = null; + pred = last; + } else { + succ = node(index); + pred = succ.prev; + } + // 遍历要加入集合的元素数组 + for (Object o : a) { + // 将元素强转成对应的类型 + @SuppressWarnings("unchecked") E e = (E) o; + Node newNode = new Node<>(pred, e, null); + if (pred == null) + // 前者为空 将此作为链首 + first = newNode; + else + // 否则就将前者的后节点指向他 + pred.next = newNode; + // pred 使用newNode + pred = newNode; + } + + if (succ == null) { + //  插入的位置为空的情况 证明最后一个就是加进去的最后一个即pred + last = pred; + } else { + // 不为空仅需将俩相连 + pred.next = succ; + succ.prev = pred; + } + + size += numNew; + modCount++; + return true; + } + + /** + * 返回在指定索引位置的节点 + * Returns the (non-null) Node at the specified element index. + */ + Node node(int index) { + // assert isElementIndex(index); + // 在前一半从前往后找 + if (index < (size >> 1)) { + Node x = first; + for (int i = 0; i < index; i++) + x = x.next; + return x; + } else { + // 在后一半从后往前找 + Node x = last; + for (int i = size - 1; i > index; i--) + x = x.prev; + return x; + } + } + ``` + +2. add + + ```java + /** + * 插入具体元素在指定的位置 移动此位置的元素和右边的元素 + * Inserts the specified element at the specified position in this list. + * Shifts the element currently at that position (if any) and any + * subsequent elements to the right (adds one to their indices). + * + * @param index index at which the specified element is to be inserted + * @param element element to be inserted + * @throws IndexOutOfBoundsException {@inheritDoc} + */ + public void add(int index, E element) { + checkPositionIndex(index); + + if (index == size) + linkLast(element); + else + linkBefore(element, node(index)); + } + ``` + +3. unlinkFirst, unlinkLast + + ```java + /** + * 移出并返回非空的第一节点f,删除头 + * Unlinks non-null first node f. + */ + private E unlinkFirst(Node f) { + // assert f == first && f != null; + final E element = f.item; + final Node next = f.next; + f.item = null; + f.next = null; // help GC + first = next; + if (next == null) + // 如果下一个为空 证明这条链就是无元素的 + last = null; + else + // 将头结点 prev至成空 + next.prev = null; + size--; + modCount++; + return element; + } + + /** + * 移出并返回非空的最后一个节点,删除尾 + * Unlinks non-null last node l. + */ + private E unlinkLast(Node l) { + // assert l == last && l != null; + final E element = l.item; + final Node prev = l.prev; + l.item = null; + l.prev = null; // help GC + last = prev; + if (prev == null) + first = null; + else + prev.next = null; + size--; + modCount++; + return element; + } + ``` + +4. linkFirst, linkLast + + ```java + /** + * 将元素作为头结点连接 + * Links e as first element. + */ + private void linkFirst(E e) { + // 头结点取出作为 新增节点的next节点 + final Node f = first; + final Node newNode = new Node<>(null, e, f); + // 将头重置成新节点 + first = newNode; + if (f == null) + // 如果头结点为空 那么尾节点也是这个新节点 + last = newNode; + else + // f的prev指向现有节点 + f.prev = newNode; + size++; + modCount++; + } + + /** + * Links e as last element. + */ + void linkLast(E e) { + // 找到之前最后一个节点 + final Node l = last; + // 组装出现有节点 + final Node newNode = new Node<>(l, e, null); + last = newNode; + if (l == null) + // 之前的最后一个节点为空 证明这条链无数据 + first = newNode; + else + // 最后一个节点的 + l.next = newNode; + size++; + modCount++; + } + ``` + +5. linkBefore + + ```java + /** + * 将元素e 插入到succ之前 + * Inserts element e before non-null Node succ. + */ + void linkBefore(E e, Node succ) { + // assert succ != null; + // succ之前的位置 + final Node pred = succ.prev; + // pred -> newNode(e) -> succ + final Node newNode = new Node<>(pred, e, succ); + succ.prev = newNode; + if (pred == null) + // 之前的最后一个节点为空 证明这条链无数据 + first = newNode; + else + pred.next = newNode; + size++; + modCount++; + } + ``` + diff --git a/week_01/05/01-ArrayList.md b/week_01/05/01-ArrayList.md new file mode 100644 index 0000000..c8cca2a --- /dev/null +++ b/week_01/05/01-ArrayList.md @@ -0,0 +1,398 @@ +# ArrayList 源码分析 + +## TOP 带着问题看源码 + +1. List list = new ArrayList(20) 扩容了几次 +2. ArrayList 怎么实现数组动态扩容,扩容时机,扩容倍数 +3. ArrayList 怎么实现remove的 +4. 为什么remove具体元素性能差 +5. ArrayList 是怎么序列化的 + +## 1. 继承和实现关系 + + + + + +- *RandomAccess 接口* + + 标记该类具有快速随机访问能力。当一个集合拥有该能力时候,采用for循环遍历会很快;若没有则采用Iterator迭代器最快。参考ArrayList的indexOf(Object o)方法和AbstractList的indexOf(Object o)方法区别。 + +- *Serializable 接口* + + 标记该类是可序列化的。 + +- *Cloneable 接口* + + 标记该类对象能够被Object.clone() + + 根据重写的clone方法实现主要分为如下两种克隆方式 + + 1. 浅克隆 + + 只copy对象本身和对象中的基本变量,不copy包含引用的对象 + + 2. 深克隆 + + 不仅copy对象本身,还copy对象包含的引用对象 + +- *AbstractList 抽象类* + + ​ 提供一些基础方法: IndexOf、clear、addAll、iterator等 + +## 2. 成员变量分析 + +```java +// 默认容量 +private static final int DEFAULT_CAPACITY = 10; +// 空数组实例(为0时候) +private static final Object[] EMPTY_ELEMENTDATA = {}; +// 默认大小时候的空数组实例 +private static final Object[] DEFAULTCAPACITY_EMPTY_ELEMENTDATA = {}; +// 存储数组 +transient Object[] elementData; +// 数组大小 +private int size; +// 数组最大容量,减8是因为可能一些VM会在数组保留一些header,防止OOM +private static final int MAX_ARRAY_SIZE = Integer.MAX_VALUE - 8; +``` + +## 3. 构造方法分析 + +### 3.1 无参构造方法 + +默认赋值一个空数组实例 + +```java +public ArrayList() { + this.elementData = DEFAULTCAPACITY_EMPTY_ELEMENTDATA; +} +``` + +### 3.2 带初始化容量的构造方法 + +可以看到是由参数的大小来创建对应大小的 elementData 数组,回到 **TOP 1** 问题,可以看出来不会发生扩容,也就是0次 + +```java +public ArrayList(int initialCapacity) { + if (initialCapacity > 0) { + this.elementData = new Object[initialCapacity]; + } else if (initialCapacity == 0) { + this.elementData = EMPTY_ELEMENTDATA; + } else { + throw new IllegalArgumentException("Illegal Capacity: "+ + initialCapacity); + } +} +``` + +### 3.3 带集合内容的构造方法 + +把传过来的集合转化为数组赋值给 elementData 数组 + +```java +public ArrayList(Collection c) { + elementData = c.toArray(); + if ((size = elementData.length) != 0) { + // c.toArray might (incorrectly) not return Object[] (see 6260652) + if (elementData.getClass() != Object[].class) + elementData = Arrays.copyOf(elementData, size, Object[].class); + } else { + // replace with empty array. + this.elementData = EMPTY_ELEMENTDATA; + } +} +``` + +## 4. 核心方法分析 + +### 4.1 获取元素 + +先 check ,再按照 index 取。check也是为了保证工程中不会出现奇奇怪怪的结果 + +```java +public E get(int index) { + rangeCheck(index); + + return elementData(index); +} +``` + +使用 final 修饰的数组来接收存储数组,对其遍历。 modCount 变量和 final 修饰的 expectedModCount 进行对比来判断是否存在并发读写情况 + +```java +public void forEach(Consumer action) { + Objects.requireNonNull(action); + final int expectedModCount = modCount; + @SuppressWarnings("unchecked") + final E[] elementData = (E[]) this.elementData; + final int size = this.size; + for (int i=0; modCount == expectedModCount && i < size; i++) { + action.accept(elementData[i]); + } + if (modCount != expectedModCount) { + throw new ConcurrentModificationException(); + } +} +``` + +### 4.2 新增元素 + +#### 4.2.1 add(E e) + +把一个元素新增到elementData,主要涉及如下几点 + +1. modCount++ 声明我新增元素了,在并发情况下起到容量是否发生变化作用 +2. 如果容量不足,则扩容数组大小(参考下面grow方法) + +```java +public boolean add(E e) { + ensureCapacityInternal(size + 1); // Increments modCount!! + elementData[size++] = e; + return true; +} +``` + +####4.2.2 add(int index, E element) + +按照index位置来插入元素,和上面方法同理。 + +```java +public void add(int index, E element) { + rangeCheckForAdd(index); + + ensureCapacityInternal(size + 1); // Increments modCount!! + System.arraycopy(elementData, index, elementData, index + 1, + size - index); + elementData[index] = element; + size++; +} +``` + +####4.2.3 grow(int minCapacity) + +第4行可以看到,使用位运算扩容了 1.5 倍大小空间,至于为啥是1.5倍,我猜是经验值。 + +回到 **TOP 2** 问题,可以明白了扩容机制是通过数组 copy方式,时机就是容量不够的时候,倍数是1.5倍 + +```java +private void grow(int minCapacity) { + // overflow-conscious code + int oldCapacity = elementData.length; + int newCapacity = oldCapacity + (oldCapacity >> 1); + if (newCapacity - minCapacity < 0) + newCapacity = minCapacity; + if (newCapacity - MAX_ARRAY_SIZE > 0) + newCapacity = hugeCapacity(minCapacity); + // minCapacity is usually close to size, so this is a win: + elementData = Arrays.copyOf(elementData, newCapacity); +} +``` + +### 4.3 更新元素 + +直接数组下标覆盖,返回旧值,至于为什么返回的是旧值,可能一方面是根据下标查询不是很影响性能索性给查询出来,另一方面下标和新值请求者都清楚也没必要返回。 + +```java +public E set(int index, E element) { + rangeCheck(index); + + E oldValue = elementData(index); + elementData[index] = element; + return oldValue; +} +``` + +### 4.4 删除元素 + +#### 4.4.1 remove(int index) + +计算要删除的下标后一位到数组末尾的长度,然后通过copy这段长度覆盖到原数组的位置,最后把最后一位置null,实现删除。 + +回到 **TOP 3** 问题,可以明白删除机制也是通过数组copy覆盖的思想来实现的 + +```java +public E remove(int index) { + rangeCheck(index); + + modCount++; + E oldValue = elementData(index); + // 计算长度 + int numMoved = size - index - 1; + if (numMoved > 0) + // param1: 源数组 + // param2: 源数组要复制的起始位置 + // param3: 目标数组 + // param4: 目标数组放置的起始位置 + // param5: 复制的长度 + System.arraycopy(elementData, index+1, elementData, index, + numMoved); + elementData[--size] = null; // clear to let GC do its work + + return oldValue; +} +``` + +####4.4.2 remove(Object o) + +首先分为两个场景,第一个是要删除的元素是null,第二个是要删除的是非null的。 + +主要是遍历要找的元素,找到该元素对应的index,然后使用 fastRemove(index) 去快速删除 + +回到 **TOP 4** 问题,可以明白计算某个元素下标的时间复杂度是 O(n) 的,所以性能没有直接根据下标删除好 + +```java +public boolean remove(Object o) { + if (o == null) { + for (int index = 0; index < size; index++) + if (elementData[index] == null) { + fastRemove(index); + return true; + } + } else { + for (int index = 0; index < size; index++) + if (o.equals(elementData[index])) { + fastRemove(index); + return true; + } + } + return false; +} +``` + +#### 4.4.3 fastRemove(int index) + +因为调用该方法都是内部计算index后调用的,所以不需要再校验index是否越界,也不需要返回oldValue。 + +```java +private void fastRemove(int index) { + modCount++; + int numMoved = size - index - 1; + if (numMoved > 0) + System.arraycopy(elementData, index+1, elementData, index, + numMoved); + elementData[--size] = null; // clear to let GC do its work +} +``` + +#### 4.4.4 clear() + +遍历赋值null,size重置为0 + +## 5. 序列化 + +首先我们在最开始就有介绍 ArrayList 类实现的有 Serializable 接口,但是我们在成员变量那一节看到的存储数组 elementData 是有 `transient` 修饰的,也就是elementData不会参与默认序列化,那实现这个 Serializable 接口还有意义么? + +其实仔细观察类里的方法你会发现有两个与序列化的流有关系的方法:`writeObject` 、`readObject` + +在序列化过程中如果有这两个方法,会默认调用这两个方法进行用户自定义的序列化和反序列化,如果没有才走默认序列化。 + +那么我们知道作者的序列化是自定义了,那为什么这样做呢,为什么不直接使用默认序列化呢? + +我们可以想下,每次扩容1.5倍,那这个数组实际会有一些空间扩容后还未被填充,如果使用默认序列化则会将null也给序列化进去。 + +接下来我们来看一下自定义序列化方法具体的实现: + +###5.1 writeObject + +写入数组大小,遍历写入数组元素,检查并发冲突 + +```java +private void writeObject(java.io.ObjectOutputStream s) + throws java.io.IOException{ + // Write out element count, and any hidden stuff + int expectedModCount = modCount; + s.defaultWriteObject(); + + // Write out size as capacity for behavioural compatibility with clone() + s.writeInt(size); + + // Write out all elements in the proper order. + for (int i=0; i

More formally, if this map contains a mapping from a key + * {@code k} to a value {@code v} such that {@code (key==null ? k==null : + * key.equals(k))}, then this method returns {@code v}; otherwise + * it returns {@code null}. (There can be at most one such mapping.) + * + *

A return value of {@code null} does not necessarily + * indicate that the map contains no mapping for the key; it's also + * possible that the map explicitly maps the key to {@code null}. + * The {@link #containsKey containsKey} operation may be used to + * distinguish these two cases. + * + * @see #put(Object, Object) + */ +public V get(Object key) { +//获取相应key的value通过getNode方法实现 + Node e; + return (e = getNode(hash(key), key)) == null ? null : e.value; +} + +/** + * Implements Map.get and related methods + * + * @param hash hash for key + * @param key the key + * @return the node, or null if none + */ +final Node getNode(int hash, Object key) { + Node[] tab; Node first, e; int n; K k; + if ((tab = table) != null && (n = tab.length) > 0 && + (first = tab[(n - 1) & hash]) != null) { + if (first.hash == hash && // always check first node + ((k = first.key) == key || (key != null && key.equals(k)))) + //判断是否为头节点 + return first; + if ((e = first.next) != null) { + if (first instanceof TreeNode) + //从红黑树里面查找 + return ((TreeNode)first).getTreeNode(hash, key); + do { + if (e.hash == hash && + ((k = e.key) == key || (key != null && key.equals(k)))) + //循环整个链表 + return e; + } while ((e = e.next) != null); + } + } + //没有则返回null + return null; +} \ No newline at end of file diff --git a/week_01/54/LinkedList-054.md b/week_01/54/LinkedList-054.md new file mode 100644 index 0000000..3a51d0f --- /dev/null +++ b/week_01/54/LinkedList-054.md @@ -0,0 +1,265 @@ +###### LinkedList基于jdk1.8源码分析 + +**成员** + +transient int size = 0;//list节点个数 + +/** + * Pointer to first node.指向头节点 + * Invariant: (first == null && last == null) || + * (first.prev == null && first.item != null) + */ +transient Node first; + +/** + * Pointer to last node.//指向尾节点 + * Invariant: (first == null && last == null) || + * (last.next == null && last.item != null) + */ +transient Node last; + +**构造函数** + +/** + * Constructs an empty list.//创建一个空list + */ +public LinkedList() { +} + +/** + * Constructs a list containing the elements of the specified + * collection, in the order they are returned by the collection's + * iterator. + * + * @param c the collection whose elements are to be placed into this list + * @throws NullPointerException if the specified collection is null + */ +public LinkedList(Collection c) { +//把一个Collection变为LinkedList + this(); + addAll(c); +} + +**核心方法** + +添加 + +/** + * Appends the specified element to the end of this list. + * + *

This method is equivalent to {@link #addLast}. + * + * @param e element to be appended to this list + * @return {@code true} (as specified by {@link Collection#add}) + */ +public boolean add(E e) { +//添加一个元素,通过linkLast方法 + linkLast(e); + return true; +} + +/** + * Links e as last element.向尾部添加一个元素 + */ +void linkLast(E e) { + final Node l = last; + final Node newNode = new Node<>(l, e, null);以尾部元素为前面节点next创建一个新节点 + last = newNode; + if (l == null) + //判断是否为空链表 + first = newNode;//头节点为newNode + else + l.next = newNode;//将原来的尾部next插入需要插入的节点 + //更新链表的次数以及大小 + size++; + modCount++; +} + +/** + * Inserts the specified element at the specified position in this list. + * Shifts the element currently at that position (if any) and any + * subsequent elements to the right (adds one to their indices). + * + * @param index index at which the specified element is to be inserted + * @param element element to be inserted + * @throws IndexOutOfBoundsException {@inheritDoc} + */ +public void add(int index, E element) { +//指定位置插入元素 + checkPositionIndex(index);//检查位置是否合理 + + if (index == size) + linkLast(element);//如果位置是在最后一位采用尾插法 + else + linkBefore(element, node(index)); +} + +/** + * Inserts element e before non-null Node succ. + */ +void linkBefore(E e, Node succ) { + // assert succ != null; + final Node pred = succ.prev;//插入位置的前节点 + final Node newNode = new Node<>(pred, e, succ);//新建节点为succ的前节点 + succ.prev = newNode;succ的前节点为newNode + if (pred == null) + //前节点为null 表示为头节点 ,直接first为newNode + first = newNode; + else + //插入位置的前节点的next为newNode + pred.next = newNode; + size++; + modCount++; +} + +/** + * Appends all of the elements in the specified collection to the end of + * this list, in the order that they are returned by the specified + * collection's iterator. The behavior of this operation is undefined if + * the specified collection is modified while the operation is in + * progress. (Note that this will occur if the specified collection is + * this list, and it's nonempty.) + * + * @param c collection containing elements to be added to this list + * @return {@code true} if this list changed as a result of the call + * @throws NullPointerException if the specified collection is null + */ +public boolean addAll(Collection c) { +//在size后面添加一个Collection集合 + return addAll(size, c); +} + +/** + * Inserts all of the elements in the specified collection into this + * list, starting at the specified position. Shifts the element + * currently at that position (if any) and any subsequent elements to + * the right (increases their indices). The new elements will appear + * in the list in the order that they are returned by the + * specified collection's iterator. + * + * @param index index at which to insert the first element + * from the specified collection + * @param c collection containing elements to be added to this list + * @return {@code true} if this list changed as a result of the call + * @throws IndexOutOfBoundsException {@inheritDoc} + * @throws NullPointerException if the specified collection is null + */ +public boolean addAll(int index, Collection c) { + checkPositionIndex(index);//检查位置是否合理 + + Object[] a = c.toArray();//转换为数组 + int numNew = a.length; + if (numNew == 0)//数组大小为0 + return false; + + Node pred, succ; + if (index == size) {//如果size和index相等,初始化succ为null,pred为尾节点 + succ = null; + pred = last; + } else { + //否则 采用折半查找index元素,并设置succ为当前index的数据,pred为当前元素的prev + succ = node(index); + pred = succ.prev; + } + + for (Object o : a) { + //循环object数组,链表循环添加 + @SuppressWarnings("unchecked") E e = (E) o; + Node newNode = new Node<>(pred, e, null); + if (pred == null) + first = newNode; + else + pred.next = newNode; + pred = newNode; + } + //循环结束后,判断如果succ是null, 说明此时是在队尾添加的,设置一下队列尾节点last, + //如果不是在队尾,则更新之前插入位置节点的前节点和当前要插入节点的尾节点 + if (succ == null) { + last = pred; + } else { + pred.next = succ; + succ.prev = pred; + } + + size += numNew; + modCount++; + return true; +} + +删除 +/** + * Removes the element at the specified position in this list. Shifts any + * subsequent elements to the left (subtracts one from their indices). + * Returns the element that was removed from the list. + * + * @param index the index of the element to be removed + * @return the element previously at the specified position + * @throws IndexOutOfBoundsException {@inheritDoc} + */ +public E remove(int index) { +//删除指定位置的元素 + checkElementIndex(index); + return unlink(node(index)); +} + +/** + * Unlinks non-null node x. + */ +E unlink(Node x) { + // assert x != null; + final E element = x.item; + final Node next = x.next;//获取当前next赋值给临时node next + final Node prev = x.prev;//获取当前prev赋值给临时node prev + + if (prev == null) { + //头节点 + first = next;//直接赋值 + } else { + prev.next = next;//把next给prev的next + x.prev = null;//设置为null + } + + if (next == null) { + //尾节点 + last = prev; + } else { + next.prev = prev; + x.next = null; + } + + x.item = null; + size--; + modCount++; + return element; +} + +查询 +/** + * Returns the element at the specified position in this list. + * + * @param index index of the element to return + * @return the element at the specified position in this list + * @throws IndexOutOfBoundsException {@inheritDoc} + */ +public E get(int index) { + checkElementIndex(index);//判断位置是否合理 + return node(index).item;//折半查找 +} + +修改 +/** + * Replaces the element at the specified position in this list with the + * specified element. + * + * @param index index of the element to replace + * @param element element to be stored at the specified position + * @return the element previously at the specified position + * @throws IndexOutOfBoundsException {@inheritDoc} + */ +public E set(int index, E element) { + checkElementIndex(index);//判断位置是否合理 + Node x = node(index);//折半查找 + E oldVal = x.item;//获取旧元素 + x.item = element;//替换成新的元素 + return oldVal;//返回旧元素 +} \ No newline at end of file diff --git a/week_01/55/ArrayList-55.md b/week_01/55/ArrayList-55.md new file mode 100644 index 0000000..7dd4011 --- /dev/null +++ b/week_01/55/ArrayList-55.md @@ -0,0 +1,230 @@ +ArrayList +定义 +1 public class ArrayList extends AbstractList +2 implements List, RandomAccess, Cloneable, java.io.Serializable +ArrayList实际上是一个动态数组,容量可以动态的增长,其继承了AbstractList,实现了List, RandomAccess, Cloneable, java.io.Serializable这些接口。 + +RandomAccess接口,被List实现之后,为List提供了随机访问功能,也就是通过下标获取元素对象的功能。 + +实现了Cloneable, java.io.Serializable意味着可以被克隆和序列化。 + +初始化 +在使用中我们经常需要new出来各种泛型的ArrayList,那么在初始化过程是怎样的呢? + +如下一行代码,创建一个ArrayList对象 + +List list = new ArrayList<>(); + + +我们来看源码,是如何初始化的,找到构造方法 + +//无参构造方法 +public ArrayList() { + super(); + this.elementData = EMPTY_ELEMENTDATA; +} + + +看到这些代码的时候,我也是不解的,elementData和EMPTY_ELEMENTDATA是什么啊?但是很明显EMPTY_ELEMENTDATA是一个常量,追本溯源我们去看一下成员属性。 + +//如果是无参构造方法创建对象的话,ArrayList的初始化长度为10,这是一个静态常量 +private static final int DEFAULT_CAPACITY = 10; +​ +//在这里可以看到我们不解的EMPTY_ELEMENTDATA实际上是一个空的对象数组 + private static final Object[] EMPTY_ELEMENTDATA = {}; +​ +//保存ArrayList数据的对象数组缓冲区 空的ArrayList的elementData = EMPTY_ELEMENTDATA 这就是为什么说ArrayList底层是数组实现的了。elementData的初始容量为10,大小会根据ArrayList容量的增长而动态的增长。 + private transient Object[] elementData; +//集合的长度 + private int size; + + + +执行完构造方法,如下图 + + + +可以说ArrayList的作者真的是很贴心,连缓存都处理好了,多次new出来的新对象,都指向同一个引用。 + +添加方法add +add(E e) + + /** + * Appends the specified element to the end of this list. + */ +//增加元素到集合的最后 +public boolean add(E e) { + ensureCapacityInternal(size + 1); // Increments modCount!! + //因为++运算符的特点 先使用后运算 这里实际上是 + //elementData[size] = e + //size+1 + elementData[size++] = e; + return true; +} + + + +先不管ensureCapacityInternal的话,这个方法就是将一个元素增加到数组的size++位置上。 + +再说回ensureCapacityInternal,它是用来扩容的,准确说是用来进行扩容检查的。下面我们来看一下整个扩容的过程 + + +//初始长度是10,size默认值0,假定添加的是第一个元素,那么minCapacity=1 这是最小容量 如果小于这个容量就会报错 +//如果底层数组就是默认数组,那么选一个大的值,就是10 +private void ensureCapacityInternal(int minCapacity) { + if (elementData == EMPTY_ELEMENTDATA) { + minCapacity = Math.max(DEFAULT_CAPACITY, minCapacity); + } + //调用另一个方法ensureExplicitCapacity + ensureExplicitCapacity(minCapacity); + } +​ + private void ensureExplicitCapacity(int minCapacity) { + //记录修改的次数 + modCount++; +​ + // overflow-conscious code + //如果传过来的值大于初始长度 执行grow方法(参数为传过来的值)扩容 + if (minCapacity - elementData.length > 0) + grow(minCapacity); + } +//真正的扩容 + private void grow(int minCapacity) { + // overflow-conscious code + int oldCapacity = elementData.length; + //新的容量是在原有的容量基础上+50% 右移一位就是二分之一 + int newCapacity = oldCapacity + (oldCapacity >> 1); + //如果新容量小于最小容量,按照最小容量进行扩容 + if (newCapacity - minCapacity < 0) + newCapacity = minCapacity; + if (newCapacity - MAX_ARRAY_SIZE > 0) + newCapacity = hugeCapacity(minCapacity); + // minCapacity is usually close to size, so this is a win: + //这里是重点 调用工具类Arrays的copyOf扩容 + elementData = Arrays.copyOf(elementData, newCapacity); + } +​ +//Arrays +public static T[] copyOf(U[] original, int newLength, Class newType) { + T[] copy = ((Object)newType == (Object)Object[].class) + ? (T[]) new Object[newLength] + : (T[]) Array.newInstance(newType.getComponentType(), newLength); + System.arraycopy(original, 0, copy, 0, + Math.min(original.length, newLength)); + return copy; +} +​ + + + +add(int index, E element) +添加到指定的位置 + + +public void add(int index, E element) { + //判断索引是否越界,如果越界就会抛出下标越界异常 + rangeCheckForAdd(index); +//扩容检查 + ensureCapacityInternal(size + 1); // Increments modCount!! + //将指定下标空出 具体作法就是index及其后的所有元素后移一位 + System.arraycopy(elementData, index, elementData, index + 1,size - index); + //将要添加元素赋值到空出来的指定下标处 + elementData[index] = element; + //长度加1 + size++; +} +//判断是否出现下标是否越界 +private void rangeCheckForAdd(int index) { + //如果下标超过了集合的尺寸 或者 小于0就是越界 + if (index > size || index < 0) + throw new IndexOutOfBoundsException(outOfBoundsMsg(index)); +} + + + +remove(int index) +ArrayList支持两种删除元素的方式 + +remove(int index) 按照下标删除 常用 + +remove(Object o) 按照元素删除 会删除和参数匹配的第一个元素 + +下面我们看一下ArrayList的实现 + + + /** + 移除list中指定位置的元素 + * Removes the element at the specified position in this list. + 所有后续元素左移 下标减1 + * Shifts any subsequent elements to the left (subtracts one from their + * indices). + *参数是要移除元素的下标 + * @param index the index of the element to be removed + 返回值是被移除的元素 + * @return the element that was removed from the list + * @throws IndexOutOfBoundsException {@inheritDoc} + */ +public E remove(int index) { + //下标越界检查 + rangeCheck(index); +//修改次数统计 + modCount++; + //获取这个下标上的值 + E oldValue = elementData(index); +//计算出需要移动的元素个数 (因为需要将后续元素左移一位 此处计算出来的是后续元素的位数) + int numMoved = size - index - 1; + //如果这个值大于0 说明后续有元素需要左移 是0说明被移除的对象就是最后一位元素 + if (numMoved > 0) + //索引index只有的所有元素左移一位 覆盖掉index位置上的元素 + System.arraycopy(elementData, index+1, elementData, index,numMoved); + // 将最后一个元素赋值为null 这样就可以被gc回收了 + elementData[--size] = null; // clear to let GC do its work +//返回index位置上的元素 + return oldValue; +} +​ +//移除特定元素 +public boolean remove(Object o) { + //如果元素是null 遍历数组移除第一个null + if (o == null) { + for (int index = 0; index < size; index++) + if (elementData[index] == null) { + //遍历找到第一个null元素的下标 调用下标移除元素的方法 + fastRemove(index); + return true; + } + } else { + //找到元素对应的下标 调用下标移除元素的方法 + for (int index = 0; index < size; index++) + if (o.equals(elementData[index])) { + fastRemove(index); + return true; + } + } + return false; +} +​ +//按照下标移除元素 +private void fastRemove(int index) { + modCount++; + int numMoved = size - index - 1; + if (numMoved > 0) + System.arraycopy(elementData, index+1, elementData, index, + numMoved); + elementData[--size] = null; // clear to let GC do its work +} + + + +ArrayList总结 +底层数组实现,使用默认构造方法初始化出来的容量是10 + +扩容的长度是在原长度基础上加二分之一 + +实现了RandomAccess接口,底层又是数组,get读取元素性能很好 + +线程不安全,所有的方法均不是同步方法也没有加锁,因此多线程下慎用 + +顺序添加很方便 + +删除和插入需要复制数组 性能很差(可以使用LinkindList) \ No newline at end of file diff --git a/week_01/55/HashMap-55.md b/week_01/55/HashMap-55.md new file mode 100644 index 0000000..7a8d7a9 --- /dev/null +++ b/week_01/55/HashMap-55.md @@ -0,0 +1,258 @@ +HashMap +定义 +public class HashMap extends AbstractMap implements Map, Cloneable, Serializable +HashMap没有什么要说的,直接切入正题,初始化一个HashMap。 + +初始化 +HashMap map = new HashMap(); +通过这个方法会调用HashMap的无参构造方法。 + +//两个常量 向下追踪 +public HashMap() { + this(DEFAULT_INITIAL_CAPACITY, DEFAULT_LOAD_FACTOR); +} + +//无参构造创建对象之后 会有两个常量 +//DEFAULT_INITIAL_CAPACITY 默认初始化容量 16 这里值得借鉴的是位运算 +static final int DEFAULT_INITIAL_CAPACITY = 1 << 4; // aka 16 +//DEFAULT_LOAD_FACTOR 负载因子默认为0.75f 负载因子和扩容有关 后文详谈 +static final float DEFAULT_LOAD_FACTOR = 0.75f; + +//最大容量为2的30次方 +static final int MAXIMUM_CAPACITY = 1 << 30; + +//以Node为元素的数组,长度必须为2的n次幂 +transient Node[] table; + +//已经储存的Node的数量,包括数组中的和链表中的,逻辑长度 +transient int size; + +threshold 决定能放入的数据量,一般情况下等于 Capacity * LoadFactor +通过上述代码我们不难发现,HashMap的底层还是数组(注意,数组会在第一次put的时候通过 resize() 函数进行分配),数组的长度为2的N次幂。 + +在HashMap中,哈希桶数组table的长度length大小必须为2的n次方(一定是合数),这是一种非常规的设计,常规的设计是把桶的大小设计为素数。相对来说素数导致冲突的概率要小于合数,Hashtable初始化桶大小为11,就是桶大小设计为素数的应用(Hashtable扩容后不能保证还是素数)。HashMap采用这种非常规设计,主要是为了在取模和扩容时做优化,同时为了减少冲突,HashMap定位哈希桶索引位置时,也加入了高位参与运算的过程。 + +那么Node是什么呢? + +//一个静态内部类 其实就是Map中元素的具体存储对象 +static class Node implements Map.Entry { + //每个储存元素key的哈希值 + final int hash; + //这就是key-value + final K key; + V value; + //next 追加的时候使用,标记链表的下一个node地址 + Node next; + + Node(int hash, K key, V value, Node next) { + this.hash = hash; + this.key = key; + this.value = value; + this.next = next; + } +此时我们就拥有了一个空的HashMap,下面我们看一下put + +put +JDK8 HashMap put的基本思路: + +对key的hashCode()进行hash后计算数组下标index; +如果当前数组table为null,进行resize()初始化; +如果没碰撞直接放到对应下标的位置上; +如果碰撞了,且节点已经存在,就替换掉 value; +如果碰撞后发现为树结构,挂载到树上。 +如果碰撞后为链表,添加到链表尾,并判断链表如果过长(大于等于TREEIFY_THRESHOLD,默认8),就把链表转换成树结构; +数据 put 后,如果数据量超过threshold,就要resize。 +public V put(K key, V value) { + //调用putVal方法 在此之前会对key做hash处理 + return putVal(hash(key), key, value, false, true); +} +//hash +static final int hash(Object key) { + int h; + // h = key.hashCode() 为第一步 取hashCode值 + // h ^ (h >>> 16) 为第二步 高位参与运算 + //具体的算法就不解释了 作用就是性能更加优良 + return (key == null) ? 0 : (h = key.hashCode()) ^ (h >>> 16); +} + +//进行添加操作 +final V putVal(int hash, K key, V value, boolean onlyIfAbsent, + boolean evict) { + Node[] tab; Node p; int n, i; + //如果当前数组table为null,进行resize()初始化 + if ((tab = table) == null || (n = tab.length) == 0) + n = (tab = resize()).length; + //(n - 1) & hash 计算出下标 如果该位置为null 说明没有碰撞就赋值到此位置 + if ((p = tab[i = (n - 1) & hash]) == null) + tab[i] = newNode(hash, key, value, null); + else { + //反之 说明碰撞了 + Node e; K k; + //判断 key是否存在 如果存在就覆盖原来的value + if (p.hash == hash && + ((k = p.key) == key || (key != null && key.equals(k)))) + e = p; + //没有存在 判断是不是红黑树 + else if (p instanceof TreeNode) + //红黑树是为了防止哈希表碰撞攻击,当链表链长度为8时,及时转成红黑树,提高map的效率 + e = ((TreeNode)p).putTreeVal(this, tab, hash, key, value); + //都不是 就是链表 + else { + for (int binCount = 0; ; ++binCount) { + if ((e = p.next) == null) { + //将next指向新的节点 + p.next = newNode(hash, key, value, null); + //这个判断是用来判断是否要转化为红黑树结构 + if (binCount >= TREEIFY_THRESHOLD - 1) // -1 for 1st + treeifyBin(tab, hash); + break; + } + // key已经存在直接覆盖value + if (e.hash == hash && + ((k = e.key) == key || (key != null && key.equals(k)))) + break; + p = e; + } + } + if (e != null) { // existing mapping for key + V oldValue = e.value; + if (!onlyIfAbsent || oldValue == null) + e.value = value; + afterNodeAccess(e); + return oldValue; + } + } + ++modCount; + if (++size > threshold) + resize(); + afterNodeInsertion(evict); + return null; +} +在刚才的代码中我们提到了红黑树是为了防止**哈希表碰撞攻击,当链表链长度为8时,及时转成红黑树,提高map的效率。**那么接下来说一说什么是哈希表碰撞攻击。 + +现在做web开发RESTful风格的接口相当的普及,因此很多的数据都是通过json来进行传递的,而json数据收到之后会被转为json对象,通常都是哈希表结构的,就是Map。 + +我们知道理想情况下哈希表插入和查找操作的时间复杂度均为O(1),任何一个数据项可以在一个与哈希表长度无关的时间内计算出一个哈希值(key),从而得到下标。但是难免出现不同的数据被定位到了同一个位置,这就导致了插入和查找操作的时间复杂度不为O(1),这就是哈希碰撞。 + +java的中解决哈希碰撞的思路是单向链表和黑红树,上文提到红黑树是JDK8之后添加,为了防止哈希表碰撞攻击,为什么?。 + +不知道你有没有设想过这样一种场景,添加的所有数据都碰撞在一起,那么这些数据就会被组织到一个链表中,随着链表越来越长,哈希表会退化为单链表。哈希表碰撞攻击就是通过精心构造数据,使得所有数据全部碰撞,人为将哈希表变成一个退化的单链表,此时哈希表各种操作的时间均提升了一个数量级,因此会消耗大量CPU资源,导致系统无法快速响应请求,从而达到拒绝服务攻击(DoS)的目的。 + +我们需要注意的是红黑树实际上并不能解决哈希表攻击问题,只是减轻影响,防护该种攻击还需要其他的手段,譬如控制POST数据的数量。 + +扩容resize() +不管是list还是map,都会遇到容量不足需要扩容的时候,但是不同于list,HashMap的扩容设计的非常巧妙,首先在上文提到过数组的长度为2的N次方,也就是说初始为16,扩容一次为32... 好处呢?就是上文提到的扩容是性能优化和减少碰撞,就是体现在此处。 + +数组下标计算: index = (table.length - 1) & hash ,由于 table.length 也就是capacity 肯定是2的N次方,使用 & 位运算意味着只是多了最高位,这样就不用重新计算 index,元素要么在原位置,要么在原位置+ oldCapacity. + +如果增加的高位为0,resize 后 index 不变;高位为1在原位置+ oldCapacity。resize 的过程中原来碰撞的节点有一部分会被分开。 + +扩容简单说有两步: + +1.扩容 + +创建一个新的Entry空数组,长度是原数组的2倍。 + +2.ReHash + +遍历原Entry数组,把所有的Entry重新Hash到新数组。 + +//HashMap的源码真的长 0.0 这段改天补上 +final Node[] resize() { + Node[] oldTab = table; + int oldCap = (oldTab == null) ? 0 : oldTab.length; + int oldThr = threshold; + int newCap, newThr = 0; + + if (oldCap > 0) { + if (oldCap >= MAXIMUM_CAPACITY) { + threshold = Integer.MAX_VALUE; + return oldTab; + } + else if ((newCap = oldCap << 1) < MAXIMUM_CAPACITY && + oldCap >= DEFAULT_INITIAL_CAPACITY) + newThr = oldThr << 1; // double threshold + } + else if (oldThr > 0) // initial capacity was placed in threshold + newCap = oldThr; + else { // zero initial threshold signifies using defaults + newCap = DEFAULT_INITIAL_CAPACITY; + newThr = (int)(DEFAULT_LOAD_FACTOR * DEFAULT_INITIAL_CAPACITY); + } + if (newThr == 0) { + float ft = (float)newCap * loadFactor; + newThr = (newCap < MAXIMUM_CAPACITY && ft < (float)MAXIMUM_CAPACITY ? + (int)ft : Integer.MAX_VALUE); + } + threshold = newThr; + @SuppressWarnings({"rawtypes","unchecked"}) + Node[] newTab = (Node[])new Node[newCap]; + table = newTab; + if (oldTab != null) { + for (int j = 0; j < oldCap; ++j) { + Node e; + if ((e = oldTab[j]) != null) { + oldTab[j] = null; + if (e.next == null) + newTab[e.hash & (newCap - 1)] = e; + else if (e instanceof TreeNode) + ((TreeNode)e).split(this, newTab, j, oldCap); + else { // preserve order + Node loHead = null, loTail = null; + Node hiHead = null, hiTail = null; + Node next; + do { + next = e.next; + if ((e.hash & oldCap) == 0) { + if (loTail == null) + loHead = e; + else + loTail.next = e; + loTail = e; + } + else { + if (hiTail == null) + hiHead = e; + else + hiTail.next = e; + hiTail = e; + } + } while ((e = next) != null); + if (loTail != null) { + loTail.next = null; + newTab[j] = loHead; + } + if (hiTail != null) { + hiTail.next = null; + newTab[j + oldCap] = hiHead; + } + } + } + } + } + return newTab; +} +为什么HashMap是线程不安全的 +由于源码过长,HashMap的其他方法就不写了。下面说一下关于HashMap的一些问题 + +1.如果多个线程同时使用put方法添加元素会丢失元素 + +假设正好存在两个put的key发生了碰撞,那么根据HashMap的实现,这两个key会添加到数组的同一个位置,这样最终就会发生其中一个线程的put的数据被覆盖。 + +2.多线程同时扩容会造成死循环 + +多线程同时检查到扩容,并且执行扩容操作,在进行rehash的时候会造成闭环链表,从而在get该位置元素的时候,程序将会进入死循环。【证明HashMap高并发下问题会在以后的文章中出现】 + +如何让HashMap实现线程安全? + +直接使用Hashtable +Collections.synchronizeMap方法 +使用ConcurrentHashMap 下篇文章就是分析ConcurrentHashMap是如何实现线程安全的 +总结 +HashMap 在第一次 put 时初始化,类似 ArrayList 在第一次 add 时分配空间。 +HashMap 的 bucket 数组大小一定是2的n次方 +HashMap 在 put 的元素数量大于 Capacity * LoadFactor(默认16 * 0.75) 之后会进行扩容 +负载因子是可以修改的,也可以大于1,但是建议不要轻易修改,除非情况非常特殊 +JDK8处于提升性能的考虑,在哈希碰撞的链表长度达到TREEIFY_THRESHOLD(默认8)后,会把该链表转变成树结构 +JDK8在 resize 的时候,通过巧妙的设计,减少了 rehash 的性能消耗 +扩容是一个特别耗性能的操作,所以当在使用HashMap的时候,估算map的大小,初始化的时候给一个大致的数值,避免map进行频繁的扩容 \ No newline at end of file diff --git a/week_01/57/ArrayList-057.txt b/week_01/57/ArrayList-057.txt new file mode 100644 index 0000000..05d656b --- /dev/null +++ b/week_01/57/ArrayList-057.txt @@ -0,0 +1,71 @@ +1. +初始化大小为10 + /** + * Default initial capacity. + */ + private static final int DEFAULT_CAPACITY = 10; + +2.构造方法: + ①:无参默认创建一个空集合 + /** + * Constructs an empty list with an initial capacity of ten. + */ + public ArrayList() { + this.elementData = DEFAULTCAPACITY_EMPTY_ELEMENTDATA; + } + ②:有参数的情况下 根据参数大小初始化集合 + public ArrayList(int initialCapacity) { + if (initialCapacity > 0) { + this.elementData = new Object[initialCapacity]; + } else if (initialCapacity == 0) { + this.elementData = EMPTY_ELEMENTDATA; + } else { + throw new IllegalArgumentException("Illegal Capacity: "+ + initialCapacity); + } + } + ③:也可以直接在构造方法中传一个集合 + public ArrayList(Collection c) { + elementData = c.toArray(); + if ((size = elementData.length) != 0) { + // c.toArray might (incorrectly) not return Object[] (see 6260652) + if (elementData.getClass() != Object[].class) + elementData = Arrays.copyOf(elementData, size, Object[].class); + } else { + // replace with empty array. + this.elementData = EMPTY_ELEMENTDATA; + } + } + +3.集合最大容量 int类型最大值减去本身储存需要8的大小 + /** + * The maximum size of array to allocate. + * Some VMs reserve some header words in an array. + * Attempts to allocate larger arrays may result in + * OutOfMemoryError: Requested array size exceeds VM limit + */ + private static final int MAX_ARRAY_SIZE = Integer.MAX_VALUE - 8; + +4.add时注意ArrayList为非线程安全的集合,多线程访问时需要注意并发修改异常(modCount不一致导致)。 + 解决方案:CopyOnWriteArrayList 写时复制、读写分离 + +5. /** + * Removes all of the elements from this list. The list will + * be empty after this call returns. + */ + public void clear() { + modCount++; + + // clear to let GC do its work + for (int i = 0; i < size; i++) + elementData[i] = null; + + size = 0; + } + 循环遍历,置空每一位 + + +其他为常见方法以及不常用方法。 + +特点:ArrayList 对比 LinkedList +底层为数组结构,查询快,增上慢 底层为链表结构,查询慢,增删快 \ No newline at end of file diff --git a/week_01/57/HashMap-057.txt b/week_01/57/HashMap-057.txt new file mode 100644 index 0000000..a894eb7 --- /dev/null +++ b/week_01/57/HashMap-057.txt @@ -0,0 +1,126 @@ +1.底层为链表+数组+红黑树 +为何用红黑树而不用其他结构? 红黑树的查询及增删效率介于链表及二叉搜索树之间,综合考虑用红黑树。 + +2.初始化大小 16 +/** + * The default initial capacity - MUST be a power of two. + */ + static final int DEFAULT_INITIAL_CAPACITY = 1 << 4; // aka 16 + +3.最大容量 +static final int MAXIMUM_CAPACITY = 1 << 30; +3.扩容因子 +static final float DEFAULT_LOAD_FACTOR = 0.75f; + +4./** + * The bin count threshold for using a tree rather than list for a + * bin. Bins are converted to trees when adding an element to a + * bin with at least this many nodes. The value must be greater + * than 2 and should be at least 8 to mesh with assumptions in + * tree removal about conversion back to plain bins upon + * shrinkage. + */ + static final int TREEIFY_THRESHOLD = 8; + + /** + * The bin count threshold for untreeifying a (split) bin during a + * resize operation. Should be less than TREEIFY_THRESHOLD, and at + * most 6 to mesh with shrinkage detection under removal. + */ + static final int UNTREEIFY_THRESHOLD = 6; + +当容量大于8时底层为数组加红黑树结构,小于6时又会转为数组加链表结构 +5.hash运算 +static final int hash(Object key) { + int h; + return (key == null) ? 0 : (h = key.hashCode()) ^ (h >>> 16); + } + +6.四个构造方法 + ①:public HashMap(int initialCapacity, float loadFactor) { + if (initialCapacity < 0) + throw new IllegalArgumentException("Illegal initial capacity: " + + initialCapacity); + if (initialCapacity > MAXIMUM_CAPACITY) + initialCapacity = MAXIMUM_CAPACITY; + if (loadFactor <= 0 || Float.isNaN(loadFactor)) + throw new IllegalArgumentException("Illegal load factor: " + + loadFactor); + this.loadFactor = loadFactor; + this.threshold = tableSizeFor(initialCapacity); + } + ②: + + public HashMap(int initialCapacity) { + this(initialCapacity, DEFAULT_LOAD_FACTOR); + } + ③: + + public HashMap() { + this.loadFactor = DEFAULT_LOAD_FACTOR; // all other fields defaulted + } + ④: + + public HashMap(Map m) { + this.loadFactor = DEFAULT_LOAD_FACTOR; + putMapEntries(m, false); + } + +7.put()方法 +final V putVal(int hash, K key, V value, boolean onlyIfAbsent, + boolean evict) { + Node[] tab; Node p; int n, i; + if ((tab = table) == null || (n = tab.length) == 0) + n = (tab = resize()).length; + if ((p = tab[i = (n - 1) & hash]) == null) + tab[i] = newNode(hash, key, value, null); + else { + Node e; K k; + if (p.hash == hash && + ((k = p.key) == key || (key != null && key.equals(k)))) + e = p; + else if (p instanceof TreeNode) + e = ((TreeNode)p).putTreeVal(this, tab, hash, key, value); + else { + for (int binCount = 0; ; ++binCount) { + if ((e = p.next) == null) { + p.next = newNode(hash, key, value, null); + if (binCount >= TREEIFY_THRESHOLD - 1) // -1 for 1st + treeifyBin(tab, hash); + break; + } + if (e.hash == hash && + ((k = e.key) == key || (key != null && key.equals(k)))) + break; + p = e; + } + } + if (e != null) { // existing mapping for key + V oldValue = e.value; + if (!onlyIfAbsent || oldValue == null) + e.value = value; + afterNodeAccess(e); + return oldValue; + } + } + ++modCount; + if (++size > threshold) + resize(); + afterNodeInsertion(evict); + return null; + } + +8.清除元素:循环置空 +public void clear() { + Node[] tab; + modCount++; + if ((tab = table) != null && size > 0) { + size = 0; + for (int i = 0; i < tab.length; ++i) + tab[i] = null; + } + } +9.非线程安全,多线程操作可能出现并发修改异常 +解决方法:ConcurrentHashMap + +其他get put 遍历等为常用方法 \ No newline at end of file diff --git a/week_01/59/week01/ArrayList.md b/week_01/59/week01/ArrayList.md new file mode 100644 index 0000000..f9bc358 --- /dev/null +++ b/week_01/59/week01/ArrayList.md @@ -0,0 +1,103 @@ +一.ArrayList概述 + 1.ArrayList是可以动态增长和缩减的索引序列,它是基于数组实现的List类 + 2.ArrayList封装了一个动态分配的Object[]数组(elementData)初始化长度是10,和一个size属性代表当前ArrayList内元素的数量。 + 3.ArrayList是线程不安全的,当多条线程访问同一个ArrayList集合时,程序需要手动保证该集合的同步性。 +二.ArrayList继承结构 + 1.ArrayList先继承AbstractList类,让AbstractList去实现下面几个类,从而减少ArrayList的代码量。 + 1.AbstractList实现List,提供一些基本的属性和方法,方便对ArrayList进行元素的操作和判断。 + 2.AbstractList实现RandomAccess,可以随机访问List元素。 + 3.AbstractList实现Cloneable,可以克隆。 + 4.AbstractList实现Serializable,可以被序列化,能够从类变成字节流传输,然后还能从字节流变成原来的类。 +三.构造方法 + 1.初始化一个长度的构造方法 + public ArrayList(int initialCapacity) { + if (initialCapacity > 0) { + this.elementData = new Object[initialCapacity]; + } else if (initialCapacity == 0) { + this.elementData = EMPTY_ELEMENTDATA; + } else { + throw new IllegalArgumentException("Illegal Capacity: "+ + initialCapacity); + } + } + 2.无参构造方法会初始化一个长度为10的list + public ArrayList() { + this.elementData = DEFAULTCAPACITY_EMPTY_ELEMENTDATA; + } + 3.传入一个集合的构造方法 + public ArrayList(Collection c) { + elementData = c.toArray();//转化成数组赋值 + if ((size = elementData.length) != 0) { + // c.toArray might (incorrectly) not return Object[] (see 6260652) + if (elementData.getClass() != Object[].class) + //每个集合的toarray()的实现方法不一样,所以需要判断一下, + //如果不是Object[].class类型,那么久需要使用ArrayList中的方法去改造一下。 + elementData = Arrays.copyOf(elementData, size, Object[].class); + } else { + // replace with empty array. + this.elementData = EMPTY_ELEMENTDATA; + } + } +四.核心方法 + 1.4个add方法 + public boolean add(E e) { //数组内加一个元素,size熟悉加1 + ensureCapacityInternal(size + 1); // Increments modCount!! + elementData[size++] = e; + return true; + } + + public void add(int index, E element) {//在一个特定的位置加一个元素 + + // 检查是否越界 + rangeCheckForAdd(index); + //检查是否需要扩容 + ensureCapacityInternal(size + 1); // Increments modCount!! + //把当前elementData的index位置到结尾长度的值,移到index+1往后的位置。空出index位置 + System.arraycopy(elementData, index, elementData, index + 1, + size - index); + elementData[index] = element; + size++; + } + public boolean addAll(Collection c) {//在当前list末尾加上一个集合 + Object[] a = c.toArray(); + int numNew = a.length; + ensureCapacityInternal(size + numNew); // Increments modCount + System.arraycopy(a, 0, elementData, size, numNew); + size += numNew; + return numNew != 0; + } + public boolean addAll(int index, Collection c) { + rangeCheckForAdd(index);//检查是否超出范围 + + Object[] a = c.toArray(); + int numNew = a.length; + //检查是否需要扩容 + ensureCapacityInternal(size + numNew); // Increments modCount + + int numMoved = size - index; + if (numMoved > 0) + System.arraycopy(elementData, index, elementData, index + numNew, + numMoved);//同理拷贝数组 + //拷贝中间数组 + System.arraycopy(a, 0, elementData, index, numNew); + size += numNew; + return numNew != 0; + } + 2.扩容数组 + private void grow(int minCapacity) { + // overflow-conscious code + int oldCapacity = elementData.length; + //右移相当于 除以2,扩大1.5倍 + int newCapacity = oldCapacity + (oldCapacity >> 1); + if (newCapacity - minCapacity < 0)//当list为空时,满足条件 + newCapacity = minCapacity; + if (newCapacity - MAX_ARRAY_SIZE > 0) //超过最大限制时 要给能给的最大值 + newCapacity = hugeCapacity(minCapacity); + // minCapacity is usually close to size, so this is a win: + elementData = Arrays.copyOf(elementData, newCapacity);//copy数组,并改变容量 + } + 3.其他核心方法类似 +五.总结 + 1.arrayList区别于数组的地方在于能够自动扩展大小,其中关键的方法就是gorw()方法,每次扩容可以扩大1.5倍。 + 2.arrayList由于本质是数组,所以它在数据的查询方面会很快,而在插入删除这些方面,性能下降很多,有移动很多数据才能达到应有的效果。 + 3.arrayList实现了RandomAccess,所以在遍历它的时候推荐使用for循环。 \ No newline at end of file diff --git a/week_01/59/week01/HashMap.md b/week_01/59/week01/HashMap.md new file mode 100644 index 0000000..34fda08 --- /dev/null +++ b/week_01/59/week01/HashMap.md @@ -0,0 +1,26 @@ +一.HashMap 概述 + HashMap是一个键对应一个值的Map集合,它在查找和插入时的效率极高,理论上可以达到O(1). + HashMap是利用hash函数对所需要存入的值进行计算存储位置。 + Hash函数会计算不通值之间的时候,会发生冲突,就叫hash冲突。 + 哈希冲突的解决方案有多种:开放定址法(发生冲突,继续寻找下一块未被占用的存储地址)(顺序寻找下一个未被占用地址,效率低), + 再散列函数法(利用值的平方,再散列寻找未被占用的地址),链地址法,而HashMap即是采用了链地址法,也就是数组+链表的方式。 +二.HashMap 继承结构 + 1.HashMap继承了AbstractMap + 2.HashMap实现了Map,(可以使用一些Map的操作) + 3.HashMAp实现了 Cloneable, (可被克隆) + 4.HashMap实现了Serializable,(可序列化) +三.构造方法 + // 1.无参构造方法、 + // 构造一个空的HashMap,初始容量为16,负载因子为0.75 + public HashMap() { + this.loadFactor = DEFAULT_LOAD_FACTOR; // all other fields defaulted + } +// 2.构造一个初始容量为initialCapacity,负载因子为0.75的空的HashMap, + public HashMap(int initialCapacity) { + this(initialCapacity, DEFAULT_LOAD_FACTOR); + } + /// 等待更新 +四.核心方法 + +五.总结 + \ No newline at end of file diff --git a/week_01/59/week01/LinkedList.md b/week_01/59/week01/LinkedList.md new file mode 100644 index 0000000..dee990d --- /dev/null +++ b/week_01/59/week01/LinkedList.md @@ -0,0 +1,111 @@ +一.LinkedList概述 + 1.LinkedList基于双向链表适用于增删频繁且查询不频繁的场景. + 2.线程不安全的且适用于单线程(这点和ArrayList很像)。 + 3.可以用LinkedList来实现栈和队列. +二.LinkedList继承结构 + 1.LinkedList继承了AbstractSequentialList + 2.LinkedList实现了List,(集合功能) + 3.LinkedList实现了Deque (双端队列功能,模拟栈和队列) + 4.LinkedList实现了Cloneable(克隆数组) + 5.LinkedList实现了Serializable(序列化) +三.构造方法 + transient int size = 0; + transient Node first; + transient Node last; + private static class Node { //内部类,定义一个节点 + E item;//当前元素的value + Node next;//下一个元素。null代表尾元素 + Node prev;//上一个元素。null代表首原素 + //定义一个节点 + Node(Node prev, E element, Node next) { + this.item = element; + this.next = next; + this.prev = prev; + } + } + //无参构造函数 + public LinkedList() { + } + //添加一个集合 + public LinkedList(Collection c) { + this(); + addAll(c); + } + +四.核心方法 + //在头部增加一个节点 + private void linkFirst(E e) { + final Node f = first; + final Node newNode = new Node<>(null, e, f); + first = newNode; + if (f == null) + last = newNode; + else + f.prev = newNode; + size++; + modCount++; + } + //在尾部增加一个节点 + void linkLast(E e) { + final Node l = last; + final Node newNode = new Node<>(l, e, null); + last = newNode; + if (l == null) + first = newNode; + else + l.next = newNode; + size++; + modCount++; + } + //在节点succ之前加入一个节点 + void linkBefore(E e, Node succ) { + // assert succ != null; + final Node pred = succ.prev; + //在pred和succ之间插入newNode + final Node newNode = new Node<>(pred, e, succ); + succ.prev = newNode; + if (pred == null) //是否是头元素 + first = newNode; + else + pred.next = newNode; + size++; //计数加1 + modCount++; + } + //删除头元素,并返回头元素的值 + private E unlinkFirst(Node f) { + // assert f == first && f != null; + final E element = f.item; + final Node next = f.next; + f.item = null; + f.next = null; // help GC + first = next; + if (next == null) + last = null; + else + next.prev = null; + size--; + modCount++; + return element; + } + //索引第index个节点 + Node node(int index) { + //assert isElementIndex(index); + if (index < (size >> 1)) { + Node x = first; + for (int i = 0; i < index; i++) + x = x.next; + return x; + } else { + Node x = last; + for (int i = size - 1; i > index; i--) + x = x.prev; + return x; + } + } + 2.其他方法类似 +五.总结 + 1.LinkedList方法内部实现是链表,且内部有fist与last指针操作数据 + 2.LinkedList线程不安全的,因为其内部添加、删除、等操作,没有进行同步操作。 + 3.LinkedList增删元素速度较快。 + 4.遍历效率(快-慢): + Iterator迭代 > for循环 \ No newline at end of file diff --git "a/week_01/60/01_ArrayList/01-ArrayList\345\205\263\347\263\273\347\261\273\345\233\276.png" "b/week_01/60/01_ArrayList/01-ArrayList\345\205\263\347\263\273\347\261\273\345\233\276.png" new file mode 100644 index 0000000..2471721 Binary files /dev/null and "b/week_01/60/01_ArrayList/01-ArrayList\345\205\263\347\263\273\347\261\273\345\233\276.png" differ diff --git "a/week_01/60/01_ArrayList/02_ArrayList\347\232\204grow\346\226\271\346\263\225\346\211\251\345\256\271\346\265\201\347\250\213.png" "b/week_01/60/01_ArrayList/02_ArrayList\347\232\204grow\346\226\271\346\263\225\346\211\251\345\256\271\346\265\201\347\250\213.png" new file mode 100644 index 0000000..881664f Binary files /dev/null and "b/week_01/60/01_ArrayList/02_ArrayList\347\232\204grow\346\226\271\346\263\225\346\211\251\345\256\271\346\265\201\347\250\213.png" differ diff --git a/week_01/60/01_ArrayList/ArrayList-060.md b/week_01/60/01_ArrayList/ArrayList-060.md new file mode 100644 index 0000000..359fa3e --- /dev/null +++ b/week_01/60/01_ArrayList/ArrayList-060.md @@ -0,0 +1,497 @@ +[TOC] + +- *本文对于`ArrayList`的源码阅读记录基于`JDK1.8`,个人浅见,如有不妥,敬请指正。* +- *文中 "`//TODO`"处代表个人暂时未懂或标示之后会学习的地方,如有高见,敬请指教。* + +# 1.ArrayList底层原理 + +## 1.1 ArrayList的UML + +![01-ArrayList关系类图](01-ArrayList关系类图.png) + +```java +public class ArrayList extends AbstractList + implements List, RandomAccess, Cloneable, java.io.Serializable{...} +``` + +- 继承了AbstractList; +- 实现了List接口; +- 实现了RandomAccess接口,支持快速随机访问,通过下标序号进行快速访问; +- 实现了Cloneable接口,支持克隆; +- 实现了Serializable接口,支持序列化; + +## 1.2 ArrayList的数据结构 + +```java +/** + * Resizable-array implementation of the List interface. Implements + * all optional list operations, and permits all elements, including + * null. +``` + +​ 如上,通过ArrayList的源码文档可知,ArrayList是可动态调整容量的数组(`Resizable-array`),ArrayList实现了List的所有操作并允许包括null在内的所有元素(ArrayList中存放的是Object[]数组,意味着ArrayList可以存放任何继承自Object的对象)。 + +## 1.3 ArrayList中定义的常量、变量以及构造函数 + +```java + //序列版本号 + private static final long serialVersionUID = 8683452581122892189L; + + /** + * 默认初始化容量 + */ + private static final int DEFAULT_CAPACITY = 10; + + /** + * 用于空实例的共享空数组实例。 + */ + private static final Object[] EMPTY_ELEMENTDATA = {}; + + /** + * JDK 1.8中添加的,也是一个空数组,与EMPTY_ELEMENTDATA区分开来 + * (当调用无参构造方法时默认复制这个空数组) + */ + private static final Object[] DEFAULTCAPACITY_EMPTY_ELEMENTDATA = {}; + + /** + * 保存添加到数组中的元素; + * elementData是一个动态数组; + * 通过public ArrayList(int initialCapacity){}构造函数时elementData=initialCapacity; + * 通过public ArrayList(){}构造函数时elementData=DEFAULT_CAPACITY=10; + */ + transient Object[] elementData; // 非私有以简化嵌套类访问 + + /** + * 数组中元素的大小 + * + * @serial + */ + private int size; + + /** + * 指定初始化容量大小的构造函数 + * + * @param initialCapacity 数组的初始化容量大小 + * @throws IllegalArgumentException 输入参数为负数时抛出异常 + */ + public ArrayList(int initialCapacity) { + if (initialCapacity > 0) { + this.elementData = new Object[initialCapacity]; + } else if (initialCapacity == 0) { + this.elementData = EMPTY_ELEMENTDATA; + } else { + throw new IllegalArgumentException("Illegal Capacity: "+ + initialCapacity); + } + } + + /** + * 无参构造函数构造,初始化容量为0 + */ + public ArrayList() { + this.elementData = DEFAULTCAPACITY_EMPTY_ELEMENTDATA; + } + + /** + * 构造一个包含指定集合的元素的列表,其顺序由集合的迭代器返回 + * + * @param c 将其元素放入此列表的集合 + * @throws NullPointerException 传入集合参数为null时抛出异常 + */ + public ArrayList(Collection c) { + elementData = c.toArray(); + if ((size = elementData.length) != 0) { + // c.toArray might (incorrectly) not return Object[] (see 6260652) + if (elementData.getClass() != Object[].class) + elementData = Arrays.copyOf(elementData, size, Object[].class); + } else { + // replace with empty array. + this.elementData = EMPTY_ELEMENTDATA; + } + } + + /** + * 数组最大容量限定; + * 预留容量空间,避免内存溢出;为什么要预留?后面会说 + */ + private static final int MAX_ARRAY_SIZE = Integer.MAX_VALUE - 8; +``` + +### 1.3.1 ArrayList的默认容量是多少? + +```java + private static final Object[] DEFAULTCAPACITY_EMPTY_ELEMENTDATA = {}; + + public ArrayList() { + this.elementData = DEFAULTCAPACITY_EMPTY_ELEMENTDATA; + } +``` + +​ 通过ArrayList的无参构造函数可以看出,初始化时ArrayList的默认容量是一个空数组,相当于默认容量为0; + +注:JDK1.8之前默认容量是10. + +### 1.3.2 elementData为什么要用transient修饰? + +​ 首先说一下transient关键字:Java的serialization提供了一种持久化对象的机制,当持久化对象时,可能有一个特殊的对象数据成员,不想用serialization机制来保存,为了在一个特定对象的一个域上关闭serialization,可以在这个域前加上关键字transient。当一个对象被序列化时,transient型变量的值不包括在序列化的表示中,非transient型的变量是被包括进去的。(HashMap、ArrayList中都使用了transient关键字) + +- transient修饰的变量不能被序列化; + +- transient只作用于实现 Serializable 接口; + +- transient只能用来修饰普通成员变量字段; + +- 不管有没有 transient 修饰,静态变量都不能被序列化; + +​ ArrayList是可序列化的类,elementdata是ArrayList用来存储元素的成员,用transient关键字修饰elementdata,真的就意味着elementdata不能序列化了吗?而且这样反序列化后的ArrayList也会把原来存储的元素弄丢? + + 如果继续看ArrayList的源码,会发现ArrayList会调用自己实现的writeObject()和readObject()方法进行序列化和反序列化,之所以这样,是因为elementdata是一个缓存数组,通常会预留一些容量,等到容量不足时再扩容,这些预留的容量空间里没有实际存储元素。所以采用transient关键字保证elementdata不会被serialization提供的持久化机制保存(序列化),再加上ArrayList自己实现的序列化和反序列方法,这样就可以保证ArrayList序列化时只会序列化实际存储的那些元素,而不包含预留容量中空的存储空间,从而节省序列化反序列化的时间和空间。 + +参考:[ArrayList中elementData为什么被transient修饰?](https://blog.csdn.net/zero__007/article/details/52166306) + +### 1.3.3 DEFAULTCAPACITY_EMPTY_ELEMENTDATA和EMPTY_ELEMENTDATA的区别? + +//TODO + +### 1.3.4 ArrayList三种构造函数 + +1.无参构造函数:无参构造函数时,创建一个容量为0的数组; + +```java + public ArrayList() { + this.elementData = DEFAULTCAPACITY_EMPTY_ELEMENTDATA; + } +``` + +```java + ArrayList arrayList = new ArrayList(); +``` + +2.指定容量构造函数 + +```java + public ArrayList(int initialCapacity) { + if (initialCapacity > 0) { + this.elementData = new Object[initialCapacity]; + } else if (initialCapacity == 0) { + this.elementData = EMPTY_ELEMENTDATA; + } else { + throw new IllegalArgumentException("Illegal Capacity: "+ + initialCapacity); + } + } +``` + +``` + ArrayList arrayList = new ArrayList(20); +``` + +- `initialCapacity > 0`时,ArrayList容量为给定的`initialCapacity`; +- `initialCapacity == 0` 时,ArrayList容量为`EMPTY_ELEMENTDATA = 0`; +- `initialCapacity < 0` 时,抛出异常:`"Illegal Capacity: "+ initialCapacity`; + +3.构造一个包含指定集合的元素的列表,按集合的迭代器返回元素的顺序排列。 + +```java +public ArrayList(Collection c) { + elementData = c.toArray(); + if ((size = elementData.length) != 0) { + // c.toArray might (incorrectly) not return Object[] (see 6260652) + if (elementData.getClass() != Object[].class) + elementData = Arrays.copyOf(elementData, size, Object[].class); + } else { + // replace with empty array. + this.elementData = EMPTY_ELEMENTDATA; + } + } +``` + +``` + ArrayList arrayList = new ArrayList(Arrays.asList(new String[]{"1","2","3"})); +``` + +1. 将传入集合`c`拷贝给`elementData`。 + +```java + elementData = c.toArray(); + | + ↓ + 根据toArray()进入Arrays查看该方法源码 + @Override + public Object[] toArray() { + return a.clone(); + } +``` + +​ 2. 对得到 `c` 拷贝后的`elementData`进行判断: + +​ 2.1传入集合为空时:指定ArrayList容量为`EMPTY_ELEMENTDATA = 0`; + +​ 2.2传入集合不为空时且类型与Object[]不相同时,进行一次`Arrays.copyOf()`,将源数组中的元素类型向上转型后将复制的新数组返回给elementData; + +### 1.3.5 ArrayList为什么规定MAX_ARRAY_SIZE = Integer.MAX_VALUE - 8 ? + +```java + /** + * The maximum size of array to allocate. + * Some VMs reserve some header words in an array. + * Attempts to allocate larger arrays may result in + * OutOfMemoryError: Requested array size exceeds VM limit + */ + private static final int MAX_ARRAY_SIZE = Integer.MAX_VALUE - 8; +``` + +​ 数组对象也是标准的JAVA对象,在JVM(HotSpot)的堆内存中的实际存储方式和格式也满足oop-class二分模型。之所以`MAX_ARRAY_SIZE = Integer.MAX_VALUE - 8` 是因为需要预留一定长度来保存对象元数据的信息,在ArrayList中预留的8字节是用来存储数组长度,因为ArrayList不能计算自己的长度。 + +​ 那为什么是`Integer.MAX_VALUE - 8`呢?是因为JVM要求对象的大小必须是8字节的整数倍,因此当“对象头+实例数据”大小不满足8字节的整数倍时,就需要补齐满足前述条件。 + +> “ +> +> 按照8字节对齐,是底层CPU数据总线读取内存数据的要求。 +> +> 通常CPU按照字长来读取数据,一个数据若不对齐则可能需要CPU读取两次; +> +> 若进行了对齐,则一次性即可读出目标数据,这将会大大节省CPU资源,因此对象大小需要补齐。 +> +> ” + +//TODO *JAVA对象在堆内存中的详情?* + +参考: + +- [Why the maximum array size of ArrayList is Integer.MAX_VALUE - 8?](https://stackoverflow.com/questions/35756277/why-the-maximum-array-size-of-arraylist-is-integer-max-value-8) +- [如何计算Java对象占堆内存中的大小 ](http://www.sohu.com/a/306845134_505779) + +# 2.ArrayList如何添加元素? + +​ 向ArrayList中添加元素,在所难免会涉及到ArrayList扩容的情况,在记录添加元素方法前先看看ArrayList的扩容机制。 + +## 2.1 ArrayList的扩容机制? + +#### 2.1.1 初始判断扩容 + +```java + /** + * 比较传入容量的大小 + * 这个方法只有在首次调用时会用到 + **/ + private static int calculateCapacity(Object[] elementData, int minCapacity) { + // 判断ArrayList是否刚初始化 + if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) { + return Math.max(DEFAULT_CAPACITY, minCapacity); + } + return minCapacity; + } + // 校验容量的大小 + private void ensureCapacityInternal(int minCapacity) { + ensureExplicitCapacity(calculateCapacity(elementData, minCapacity)); + } + + private void ensureExplicitCapacity(int minCapacity) { + modCount++;//记录修改,modCount的作用是在List迭代器遍历时用作线程安全检查 + + // 如果传入的值大于初始长度,则调用grow()方法进行扩容 + if (minCapacity - elementData.length > 0) + grow(minCapacity); + } +``` + +- 初始化判断会判断ArrayList是否刚初始化,如果成立,则将ArrayList扩容为默认容量`DEFAULT_CAPACITY = 10`; +- 若判断 `minCapacity` 和 `DEFAULT_CAPACITY` 的大小,在二者的最大值内还有可用空间,则取二者中的最大值为ArrayList的容量; +- 当`minCapacity` 和`DEFAULT_CAPACITY`的最大值容量都没有可用空间了,则调用`grow()`进行扩容; + +#### 2.1.2 调用`grow()`扩容 + +```java + /** + * 对列表进行扩容,以确保它至少可以容纳最小容量参数指定的元素。 + * + * @param minCapacity 传入所需的最小容量 + */ + private void grow(int minCapacity) { + // 记录原容量的大小 + int oldCapacity = elementData.length; + // 设置新容量,新容量大小为原容量的1.5倍(右移一位就是1/2) + int newCapacity = oldCapacity + (oldCapacity >> 1); + // 判断新容量是否满足需求,若扩容后仍不够用,将传入最小容量大小复制给新容量 + if (newCapacity - minCapacity < 0) + newCapacity = minCapacity; + // 判断新容量是否超出最大长度限制,如果超出,进入hugeCapacity()处理 + if (newCapacity - MAX_ARRAY_SIZE > 0) + newCapacity = hugeCapacity(minCapacity); + // 将原数组的数据复制至新数组, ArrayList的底层数组引用指向新数组 + elementData = Arrays.copyOf(elementData, newCapacity); + } + + private static int hugeCapacity(int minCapacity) { + if (minCapacity < 0) // overflow + throw new OutOfMemoryError(); + return (minCapacity > MAX_ARRAY_SIZE) ? + Integer.MAX_VALUE : + MAX_ARRAY_SIZE; + } +``` + +- ArrayList调用grow()扩容后容量变为原来1.5倍; + +![02_ArrayList的grow方法扩容流程](02_ArrayList的grow方法扩容流程.png) + +//TODO *`Arrays.copyOf()`详情?* + +## 2.2 将指定元素添加到列表的尾部 + +```java + /** + * Appends the specified element to the end of this list. + * + * @param e element to be appended to this list + * @return true (as specified by {@link Collection#add}) + */ + public boolean add(E e) { + ensureCapacityInternal(size + 1); // Increments modCount!! + elementData[size++] = e; + return true; + } +``` + +​ 按照前文的ArrayList初始化扩容,的到一个容量为10的ArrayList集合。 + +## 2.3 将指定元素添加到列表的指定位置 + +```java + /** + * 将指定的元素插入指定位置中。 + * 将当前在该位置的元素(如果有的话)和任何后续元素向右移动(向其索引添加一个) + * @param index 指定元素插入的索引 + * @param element 指定元素 + * @throws IndexOutOfBoundsException {@inheritDoc} + */ + public void add(int index, E element) { + rangeCheckForAdd(index);//判断索引是否异常 + + ensureCapacityInternal(size + 1); // 判断是否需要扩容 + //将指定下标空出 具体作法就是index及其后的所有元素后移一位 + System.arraycopy(elementData, index, elementData, index + 1, + size - index); + // 将要添加元素赋值到空出来的指定索引处 + elementData[index] = element; + // 长度+1 + size++; + } + + /** + * A version of rangeCheck used by add and addAll. + * add和addAll使用的rangeCheck版本。 + */ + private void rangeCheckForAdd(int index) { + if (index > size || index < 0) + throw new IndexOutOfBoundsException(outOfBoundsMsg(index)); + } +``` + +​ 值得注意的是,对索引处及其后的元素使用了`System.arraycopy()`方法,该方法源码如下: + +```java + /** + * 从指定的源数组开始复制一个数组,该数组从指定位置开始复制到目标数组的指定位置。 + * Object src 源数组对象 + * int srcPos 源数组的起始位置 + * Object dest 目标数组对象 + * int destPos 目标数组的起始位置 + * int length 要复制的元素数量 + **/ + public static native void arraycopy(Object src, int srcPos, + Object dest, int destPos, + int length); +``` + +​ 该方法是一个数组拷贝的方法,使用`native`修饰,一般是由其它语言实现的(如C、C++)。 + +# 3.ArrayList的数组拷贝是怎么实现的? + +```java + /** + * Returns a shallow copy of this ArrayList instance. (The + * elements themselves are not copied.) + * + * @return a clone of this ArrayList instance + */ + public Object clone() { + try { + ArrayList v = (ArrayList) super.clone(); + v.elementData = Arrays.copyOf(elementData, size);//传入源数组对象、目标数组长度 + v.modCount = 0; + return v; + } catch (CloneNotSupportedException e) { + // this shouldn't happen, since we are Cloneable + throw new InternalError(e); + } + } +``` + +​ 从源码可看出其实ArrayList的拷贝最关键是调用了`Arrays.copyOf()`方法,该方法源码如下: + +```java + public static T[] copyOf(T[] original, int newLength) { + return (T[]) copyOf(original, newLength, original.getClass()); + } + + public static T[] copyOf(U[] original, int newLength, Class newType) { + @SuppressWarnings("unchecked") + T[] copy = ((Object)newType == (Object)Object[].class) + ? (T[]) new Object[newLength] + : (T[]) Array.newInstance(newType.getComponentType(), newLength); + System.arraycopy(original, 0, copy, 0, + Math.min(original.length, newLength)); + return copy; + } +``` + +- 首先,创建一个和**源数组元素**相同类型的新数组; +- 然后调用`System.arraycopy()`,并且在该方法最后一个参数(要复制的元素数量)中对**源数组长度**和**目标数组长度**进行取最小值判断; + +## 3.1 ArrayList中的拷贝是深拷贝还是浅拷贝呢? + +```java + ArrayList v = (ArrayList) super.clone(); +``` + +​ 是浅拷贝:在ArrayList的clone()方法中,第一句可以看到调用了`super.clone()`方法复制了一个全新的对象并将其赋值给`V`。由于`java.lang.Object.clone()`只是一种浅拷贝,所以V的`elementData`引用的还是当前ArrayList的`elementData`引用,这就意味着在`V`上进行操作,会影响原来的ArrayList值。 + +## 3.2 如何复制某个ArrayList到另一个ArrayList中去? + +- 浅拷贝 + - 调用`ArrayList.clone()`; + - 使用ArrayList构造方法; + +```java + ArrayList newArray = oldArray.clone(); + + ArrayList newArray = new ArrayList(oldArray); +``` + +- 深拷贝 + - 使用`Collection.copy()`; + + + + + + + + + + + + + + + + + + + + + + + diff --git "a/week_01/60/02_HashMap/02-HashMap\345\205\263\347\263\273\347\261\273\345\233\276.png" "b/week_01/60/02_HashMap/02-HashMap\345\205\263\347\263\273\347\261\273\345\233\276.png" new file mode 100644 index 0000000..4d76ab6 Binary files /dev/null and "b/week_01/60/02_HashMap/02-HashMap\345\205\263\347\263\273\347\261\273\345\233\276.png" differ diff --git "a/week_01/60/02_HashMap/02-HashMap\346\225\260\346\215\256\347\273\223\346\236\204.png" "b/week_01/60/02_HashMap/02-HashMap\346\225\260\346\215\256\347\273\223\346\236\204.png" new file mode 100644 index 0000000..b4f3c9e Binary files /dev/null and "b/week_01/60/02_HashMap/02-HashMap\346\225\260\346\215\256\347\273\223\346\236\204.png" differ diff --git a/week_01/60/02_HashMap/HashMap-060.md b/week_01/60/02_HashMap/HashMap-060.md new file mode 100644 index 0000000..a74d2fe --- /dev/null +++ b/week_01/60/02_HashMap/HashMap-060.md @@ -0,0 +1,188 @@ +[TOC] + + + +# 1.HashMap底层原理 + +## 1.1 HashMap的UML + +![HashMap的UML](02-HashMap关系类图.png) + +- HashMap继承了AbstractMap; +- HashMap实现了Cloneable接口,支持克隆; +- HashMap实现了Serializable接口,支持序列化; + +## 1.2 HashMap的数据结构 + +​ HashMap底层的数据是数组,被称为哈希桶,每个桶存放的是链表,链表中的每个节点,就是HashMap中的每个元素。综合起来HashMap数据结构=**数组+链表+红黑树**。 + +![02-HashMap数据结构](02-HashMap数据结构.png) + +### 1.2.1 为什么用数组+链表? + +- **数组**用来确定元素所在桶位置,利用元素的key的hash值对数组长度取模得到。 +- **链表**用来解决hash冲突,当出现hash值一样的情形时,在数组对应位置形成一条链表。 + +## 1.3 HashMap源码中的常量、变量 + +```java +/** + * 默认初始容量(必须是2的幂) + */ + static final int DEFAULT_INITIAL_CAPACITY = 1 << 4; // aka 16 + + /** + * 最大容量(2的30次方) + */ + static final int MAXIMUM_CAPACITY = 1 << 30; + + /** + * 默认加载因子 + */ + static final float DEFAULT_LOAD_FACTOR = 0.75f; + + /** + * 链表转换为红黑树的阈值 + */ + static final int TREEIFY_THRESHOLD = 8; + + /** + * 树的链表还原阈值 + */ + static final int UNTREEIFY_THRESHOLD = 6; + + /** + * 哈希表的最小树形化容量 + */ + static final int MIN_TREEIFY_CAPACITY = 64; +``` + +### 1.3.1 HashMap默认初始化容量(DEFAULT_INITIAL_CAPACITY)为什么是2的幂? + +​ 在HashMap的存取方法中,需要对元素所在数组索引位置进行定位,在`put()`方法中有`i = (n - 1) & hash`,在`get()`方法中有`(n - 1) & hash`(put()、get()详情在后面),索引公式为`(n - 1) & hash`,当初始化容量n大小是2的幂时,索引公式等价于`n%hash`。定位下标一般用取余法,这里为什么不用取余呢? + +- 与运算(&)比取余(%)运算效率高; + +- 求余运算: a % b就相当与a-(a / b)*b 的运算; + +- 与运算: 一个指令就搞定 + + 所以,默认初始化容量大小定义为16,也就是2的幂,是为了使用更高效的与运算。 + +​ 还有一个原因就是在进行hash之后的位运算时,2的幂能保证计算的位数都是1,这样最终的下标就完全由key的hash值来决定也就更好的解决了hash碰撞。 + +### 1.3.2 HashMap默认初始化容量(DEFAULT_INITIAL_CAPACITY)为什么是16? + +​ 未论证:之所以取16是对效率和内存的一个平衡,如果太小,扩容会更频繁,如果太大,又会多占用内存空间。 + +参考:[Choosing the hash map's capacity](https://pzemtsov.github.io/2015/12/14/choosing-the-hash-maps-capacity.html) + +### 1.3.3 HashMap的最大容量(MAXIMUM_CAPACITY)为什么是1 << 30 ? + +​ int占4字节*一个字节占8位=32位整型,在二进制数中,最左边一位是符号位,用来表示正负,所以HashMap最大容量是1 << 30,而不是31次方或32次方。 + +### 1.3.4 HashMap的默认加载因子(DEFAULT_LOAD_FACTOR)为什么是0.75f ? + +```java + *

As a general rule, the default load factor (.75) offers a good + * tradeoff between time and space costs. Higher values decrease the + * space overhead but increase the lookup cost (reflected in most of + * the operations of the HashMap class, including + * get and put). The expected number of entries in + * the map and its load factor should be taken into account when + * setting its initial capacity, so as to minimize the number of + * rehash operations. If the initial capacity is greater than the + * maximum number of entries divided by the load factor, no rehash + * operations will ever occur. + * +译文:通常,默认负载因子(.75)在时间和空间成本之间提供了一个很好的折衷方案。 较高的值会减少空间开销,但会增加查找成本(在HashMap类的大多数操作中都得到体现,包括get和put)。 设置映射表的初始容量时,应考虑映射中的预期条目数及其负载因子,以最大程度地减少重新哈希操作的数量。 如果初始容量大于最大条目数除以负载因子,则将不会进行任何哈希操作。 + +``` + +​ 加载因子表示哈希表的填满程度,和HashMap的扩容紧密相关,根据HashMap源码中的文档(如上)可知,之所以选择0.75f作为默认加载因子,是`between time and space`在时间成本和空间成本上取的一个相对折中的方案。 + +​ 有一种解释是根据数学公式来推测的,一个桶空和非空的概率为0.5,通过牛顿二项式等数学计算,得到这个加载因子的值为`log(2)≈0.693`,而最后选择0.75也许是因为0.75是较为接近这个结果且和默认容量相乘为一个整数12(1`6*0.75=12`) + +​ 下面这个假设推导大佬们可以看看: + +`让s代表大小,n代表增加的键数。使用二项式定理,存储桶为空的概率为:` +$$ +P(0) = C(n, 0) * (1/s)^0 * (1 - 1/s)^(n - 0) +$$ +`因此,如果少于` +$$ +log(2)/log(s/(s - 1)) keys +$$ +`随着s达到无穷大,并且如果添加的键数达到P(0)= .5,则n / s迅速接近log(2):` +$$ +lim (log(2)/log(s/(s - 1)))/s as s -> infinity = log(2) ~ 0.693... +$$ +参考:[What is the significance of load factor in HashMap?](https://stackoverflow.com/questions/10901752/what-is-the-significance-of-load-factor-in-hashmap) + +### 1.3.5 链表转换红黑树的阀值(TREEIFY_THRESHOLD)为什么是8? + +```java + * Because TreeNodes are about twice the size of regular nodes, we + * use them only when bins contain enough nodes to warrant use + * (see TREEIFY_THRESHOLD). And when they become too small (due to + * removal or resizing) they are converted back to plain bins. In + * usages with well-distributed user hashCodes, tree bins are + * rarely used. Ideally, under random hashCodes, the frequency of + * nodes in bins follows a Poisson distribution + * (http://en.wikipedia.org/wiki/Poisson_distribution) with a + * parameter of about 0.5 on average for the default resizing + * threshold of 0.75, although with a large variance because of + * resizing granularity. Ignoring variance, the expected + * occurrences of list size k are (exp(-0.5) * pow(0.5, k) / + * factorial(k)). The first values are: + * + * 0: 0.60653066 + * 1: 0.30326533 + * 2: 0.07581633 + * 3: 0.01263606 + * 4: 0.00157952 + * 5: 0.00015795 + * 6: 0.00001316 + * 7: 0.00000094 + * 8: 0.00000006 + * more: less than 1 in ten million +``` + +`“` + +​ 理想情况下,在随机哈希码情况下,对于随机0.75的加载因子,桶中节点的分布频率服从参数为0.5的泊松分布,即使粒度调整会产生较大方差。 + +​ 由源码中给出如上的参照表可知,当链表中元素个数为8时概率已经非常小了,所以链表转换红黑树的阈值选择了8。 + +`”` + +### 1.3.6 一个树的链表还原阈值(UNTREEIFY_THRESHOLD)为什么是6? + +​ 为了防止链表和树之间频繁的转换,如果是7的话,假设一个HashMap在临界点7频繁的插入和删除元素,链表中元素个数始终在`[7,8]`中徘徊,就会造成树和链表频繁的转换,降低HashMap操作的效率。 + +### 1.3.7 哈希表的最小树形化容量(MIN_TREEIFY_CAPACITY)为什么是64? + +​ 容量低于64时,哈希碰撞的几率比较大,而这个时候出现长链表的可能性会稍微大一些,这种原因下产生的长链表,应该优先选择扩容而避免不必要的数化。 + +参考:[面试加分项-HashMap源码中这些常量的设计目的-FROM掘金-Jay_huaxiao](https://juejin.im/post/5d7195f9f265da03a6533942#heading-0) + + + + + + + + + + + + + + + + + + + + + diff --git "a/week_01/61/ArrayList\346\272\220\347\240\201\351\230\205\350\257\273.md" "b/week_01/61/ArrayList\346\272\220\347\240\201\351\230\205\350\257\273.md" new file mode 100644 index 0000000..8684250 --- /dev/null +++ "b/week_01/61/ArrayList\346\272\220\347\240\201\351\230\205\350\257\273.md" @@ -0,0 +1,244 @@ +#

ArrayList源码分析
+ +- 以下分析基于JDK1.8 + +## ArrayList 简介 + + ArrayList是可以动态扩容和动态删除冗余容量的索引序列,基于数组实现的集合,是常用的Java集合之一。 + +``` + ArrayLis继承自抽象类AbstractList实现了List接口等 +``` + +和Vector不同,ArrayList中的操作不是线程安全的!所以,建议在单线程中才使用ArrayList,而在多线程中可以选择Vector或者CopyOnWriteArrayList,或者使用Collections工具类的synchronizedList方法将其包装 + +## ArrayList分析 + +1、属性分析 + +- ```java + // 序列号ID,ArrayList实现了Serializable接口因此是可在网络中传输 + private static final long serialVersionUID = 8683452581122892189L; + ``` + +- ```java + // //默认初始容量 + private static final int DEFAULT_CAPACITY = 10; + ``` + +- ```java + // 一个空对,当用户指定ArrayList容量为0时,返回该数组 + private static final Object[] EMPTY_ELEMENTDATA = {}; + ``` + +- ```java + // 一个空对象,如果使用默认构造函数创建,则默认对象内容默认是该值 + private static final Object[] DEFAULTCAPACITY_EMPTY_ELEMENTDATA = {}; + ``` + +- ```java + // 动态数组的实际大小 ,默认为0 + private int size; + ``` + +- ```java + // 最大数组容量Integer -8 + private static final int MAX_ARRAY_SIZE = Integer.MAX_VALUE - 8; + ``` + + + +2、构造方法分析 + +```java +// 传入初始容量 initialCapacity +public ArrayList(int initialCapacity) { + if (initialCapacity > 0) { + this.elementData = new Object[initialCapacity]; + } else if (initialCapacity == 0) { + this.elementData = EMPTY_ELEMENTDATA; + } else { + throw new IllegalArgumentException("Illegal Capacity: "+ + initialCapacity); + } + } + + //无参构造 DEFAULTCAPACITY_EMPTY_ELEMENTDATA=={} 注意此时初始容量是0,而不是的 10 +//当元素第一次被加入时,扩容至默认容量 10 + public ArrayList() { + this.elementData = DEFAULTCAPACITY_EMPTY_ELEMENTDATA; + } + + // 创建一个包含collection或其子类的ArrayList 要放入 ArrayList 中的集合,其内元素将会全部添加到新建的 ArrayList 实例中 + public ArrayList(Collection c) { + // 集合传化成Object[]数组 + elementData = c.toArray(); + // 转化后的数组长度赋给当前ArrayList的size,并判断是否为0 + if ((size = elementData.length) != 0) { + // c.toArray might (incorrectly) not return Object[] (see 6260652) + if (elementData.getClass() != Object[].class) + // 若 c.toArray() 返回的数组类型不是 Object[],则利用 Arrays.copyOf(); 来构造一个大小为 size 的 Object[] 数组 + elementData = Arrays.copyOf(elementData, size, Object[].class); + } else { + // replace with empty array. + // 替换空数组 + this.elementData = EMPTY_ELEMENTDATA; + } + } + + +``` + +3、常见方法分析 + +- add方法 + +```java +public boolean add(E e) { + // 赋值初始长度或者扩容,新增元素,当前实际size+1的长度 + ensureCapacityInternal(size + 1); // Increments modCount!! + elementData[size++] = e; + return true; + } +// 确保elemenData数组有合适的大小- 如果元素为空,则复制长度默认为10 或者更大 + private void ensureCapacityInternal(int minCapacity) { + if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) { + minCapacity = Math.max(DEFAULT_CAPACITY, minCapacity); + } + + ensureExplicitCapacity(minCapacity); + } +// 保证elemenData数组有合适的大小 +private void ensureExplicitCapacity(int minCapacity) { + //记录修改次数,迭代中不一致会触发fail-fast机制,因此在遍历中删除元素的正确做法应该是使用Iterator.remove() + modCount++; + + // overflow-conscious code + if (minCapacity - elementData.length > 0) + grow(minCapacity); + } +// 扩容 + private void grow(int minCapacity) { + // overflow-conscious code + int oldCapacity = elementData.length; + int newCapacity = oldCapacity + (oldCapacity >> 1); + if (newCapacity - minCapacity < 0) + newCapacity = minCapacity; + if (newCapacity - MAX_ARRAY_SIZE > 0) + newCapacity = hugeCapacity(minCapacity); + //拷贝扩容 + elementData = Arrays.copyOf(elementData, newCapacity); + } + // 如果小于0 就报错,如果大于最大值 则取最大值 + private static int hugeCapacity(int minCapacity) { + if (minCapacity < 0) // overflow + throw new OutOfMemoryError(); + return (minCapacity > MAX_ARRAY_SIZE) ? + Integer.MAX_VALUE : + MAX_ARRAY_SIZE; + } + + + // 给指定下标,添加元素 + public void add(int index, E element) { + rangeCheckForAdd(index);//检查是否越界 + //赋值初始长度 或者扩容 + ensureCapacityInternal(size + 1); // Increments modCount!! + System.arraycopy(elementData, index, elementData, index + 1, + size - index); + elementData[index] = element; + size++; + } + +``` + +- remove方法 + + ```java + //根据指定下标 删除元素 + public E remove(int index) { + rangeCheck(index);//检查越界 + + modCount++; + E oldValue = elementData(index); + //将数组elementData中index位置之后的所有元素向前移一位 + int numMoved = size - index - 1; + if (numMoved > 0) + System.arraycopy(elementData, index+1, elementData, index, + numMoved); + elementData[--size] = null; // clear to let GC do its work 将原数组最后一个位置置为null,由GC回收 + + return oldValue; + } + + // 根据指定元素 删除元素  + public boolean remove(Object o) { + // ArrayList中允许存放null,因此下面通过两种情况来分别处理。 + if (o == null) { + for (int index = 0; index < size; index++) + if (elementData[index] == null) { + // 私有的移除方法,跳过index参数的边界检查以及不返回任何值 + fastRemove(index); + return true; + } + } else { + for (int index = 0; index < size; index++) + if (o.equals(elementData[index])) { + fastRemove(index); + return true; + } + } + return false; + } + + // 根据下标快速删除元素 + private void fastRemove(int index) { + modCount++; + //将数组elementData中index位置之后的所有元素向前移一位 + int numMoved = size - index - 1; + if (numMoved > 0) + System.arraycopy(elementData, index+1, elementData, index, + numMoved); + elementData[--size] = null; // clear to let GC do its work + } + ``` + +- set方法 + + ```java + public E set(int index, E element) { + rangeCheck(index);//越界检查 + + E oldValue = elementData(index); + elementData[index] = element; // 指定位置替换 + return oldValue; + } + ``` + +- get方法 + + ```java + public E get(int index) { + rangeCheck(index); + + return elementData(index); + } + E elementData(int index) { + return (E) elementData[index];//取数组指定位置并返回 + } + ``` + +## 总结 + +和hashmap一样 扩容都会增加时耗初始化的时候指定合适的长度。 + +arrayList由于本质是数组,所以它在数据的查询方面会很快,而在插入删除这些方面,性能下降很多,有移动很多数据才能达到应有的效果,而LinkedList则相反 + + + + + + + + + diff --git "a/week_01/61/HashMap\346\272\220\347\240\201\351\230\205\350\257\273.md" "b/week_01/61/HashMap\346\272\220\347\240\201\351\230\205\350\257\273.md" new file mode 100644 index 0000000..c88bc3a --- /dev/null +++ "b/week_01/61/HashMap\346\272\220\347\240\201\351\230\205\350\257\273.md" @@ -0,0 +1,312 @@ +#
HashMap源码分析
+ +- 以下分析基于JDK1.8 + +## HashMap 简介 + +HashMap 主要用来存放键值对,它基于哈希表的Map接口实现,是常用的Java集合之一。 + +``` +hashMap继承自抽象类AbstractMap实现了Map接口等 +``` + +HashMap 最多允许一条记录的键为null,允许多条记录的值为null。HashMap 是非线程安全的,即任一时刻有多个线程同时写 HashMap ,可能会导致数据不一致,。如果要满足线程安全,可以使用 Collections 的 SynchronizedMap 方法 或者使用 ConcurrentHashMap。 + +## HashMap分析 + +1、属性分析 + +- ```java + // 序列号ID,hashmap实现了Serializable接口因此是可在网络中传输 + private static final long serialVersionUID = 362498820763181265L; + ``` + +- ```java + // 默认的初始容量是16 + static final int DEFAULT_INITIAL_CAPACITY = 1 << 4; // aka 16 + ``` + +- ```java + // 最大容量 + static final int MAXIMUM_CAPACITY = 1 << 30; + ``` + +- ``` + // 默认的负载率 75% + static final float DEFAULT_LOAD_FACTOR = 0.75f; + ``` + +- ```java + // 当node上的结点数大于这个值时会由链表结构转成红黑树 + static final int TREEIFY_THRESHOLD = 8; + ``` + +- ```java + // 当node上的结点数小于这个值时树转链表 + static final int UNTREEIFY_THRESHOLD = 6; + ``` + +- ```java + // 桶中结构转化为红黑树对应的table的最小大小 + static final int MIN_TREEIFY_CAPACITY = 64; + ``` + +2、构造方法分析 + +```java +// 无参构造函数 + public HashMap() { + this.loadFactor = DEFAULT_LOAD_FACTOR; // all other fields defaulted + } +// 指定“容量大小”的构造函数 + public HashMap(int initialCapacity) { + this(initialCapacity, DEFAULT_LOAD_FACTOR); + } +// 含另一个“Map”的构造函数 + public HashMap(Map m) { + this.loadFactor = DEFAULT_LOAD_FACTOR; + putMapEntries(m, false); + } +// 指定“容量大小”和“加载因子”的构造函数 + public HashMap(int initialCapacity, float loadFactor) { + // 初始容量小于0,则抛出异常 + if (initialCapacity < 0) + throw new IllegalArgumentException("Illegal initial capacity: " + + initialCapacity); + // 始容量大于容量最大值,则使用最大值作为初始容量 + if (initialCapacity > MAXIMUM_CAPACITY) + initialCapacity = MAXIMUM_CAPACITY; + // 果负载率小于等于0或负载率不是浮点数,则抛出异常 + if (loadFactor <= 0 || Float.isNaN(loadFactor)) + throw new IllegalArgumentException("Illegal load factor: " + + loadFactor); + this.loadFactor = loadFactor; + // 设置阀值为初始容量 + this.threshold = tableSizeFor(initialCapacity); + + } +// 回大于输入参数且最近的2的整数次幂的数。比如10,则返回16 相比于1.7提升效率 + static final int tableSizeFor(int cap) { + int n = cap - 1; + n |= n >>> 1; + n |= n >>> 2; + n |= n >>> 4; + n |= n >>> 8; + n |= n >>> 16; + return (n < 0) ? 1 : (n >= MAXIMUM_CAPACITY) ? MAXIMUM_CAPACITY : n + 1; + } + + +``` + +3、常见方法分析 + +- get方法 + +```java +public V get(Object key) { + Node e; + + return (e = getNode(hash(key), key)) == null ? null : e.value; +} +//计算key的hash值 +static final int hash(Object key) { + int h; + return (key == null) ? 0 : (h = key.hashCode()) ^ (h >>> 16); + } + +final Node getNode(int hash, Object key) { + Node[] tab; Node first, e; int n; K k; + if ((tab = table) != null && (n = tab.length) > 0 && + (first = tab[(n - 1) & hash]) != null) {// 首节点不为空 + // 数组元素相等 + if (first.hash == hash && // always check first node + ((k = first.key) == key || (key != null && key.equals(k)))) + return first; + // 桶中不止一个节点 + if ((e = first.next) != null) { + // 首节点是红黑树按红黑树算法找节点 + if (first instanceof TreeNode) + return ((TreeNode)first).getTreeNode(hash, key); + do { + // 首节点是链表按链表算法找节点 + if (e.hash == hash && + ((k = e.key) == key || (key != null && key.equals(k)))) + return e; + } while ((e = e.next) != null); + } + } + return null; + } +``` + +- put方法 + + ```java + public V put(K key, V value) { + return putVal(hash(key), key, value, false, true); + } + + /** + * Implements Map.put and related methods + * + * @param hash hash for key + * @param key the key + * @param value the value to put + * @param onlyIfAbsent if true, don't change existing value + * @param evict if false, the table is in creation mode. + * @return previous value, or null if none + */ + final V putVal(int hash, K key, V value, boolean onlyIfAbsent, + boolean evict) { + Node[] tab; Node p; int n, i; + // table未初始化或者长度为0,进行扩容创建table + if ((tab = table) == null || (n = tab.length) == 0) + n = (tab = resize()).length;// 桶中已经存在元素 + // (n - 1) & hash 确定元素存放在哪个桶中,桶为空,新生成结点放入桶里 + if ((p = tab[i = (n - 1) & hash]) == null) + //此位置没有对象创建新的node + tab[i] = newNode(hash, key, value, null); + else {// 桶中已经存在元素 + Node e; K k; + // 比较桶中第一个元素(数组中的结点)的hash值相等,key相等 + if (p.hash == hash && + ((k = p.key) == key || (key != null && key.equals(k)))) + e = p; + // hash值不相等,即key不相等;为红黑树结点 + else if (p instanceof TreeNode) + e = ((TreeNode)p).putTreeVal(this, tab, hash, key, value); + else {//链表 + // 在链表最末插入结点 + for (int binCount = 0; ; ++binCount) { + // 是否到达表表尾部 + if ((e = p.next) == null) { + // 在尾部插入新结点 + p.next = newNode(hash, key, value, null); + // 判断是否需要转换为红黑树 + if (binCount >= TREEIFY_THRESHOLD - 1) // -1 for 1st + treeifyBin(tab, hash); + break; + } + // 判断链表中结点的key值与插入的元素的key值是否相等 + if (e.hash == hash && + ((k = e.key) == key || (key != null && key.equals(k)))) + break; + p = e; + } + } + // 表示在桶中找到key值hash值与插入元素相等的结点 + if (e != null) { // existing mapping for key + V oldValue = e.value; + if (!onlyIfAbsent || oldValue == null) + // 替换 + e.value = value; + afterNodeAccess(e); + return oldValue; + } + } + ++modCount; + + if (++size > threshold)// 实际大小大于阈值则扩容 + resize(); + afterNodeInsertion(evict); + return null; + } + + + // 扩容 + final Node[] resize() { + Node[] oldTab = table; + int oldCap = (oldTab == null) ? 0 : oldTab.length; + int oldThr = threshold; + int newCap, newThr = 0; + if (oldCap > 0) { + // 超过最大值就不再扩了, + if (oldCap >= MAXIMUM_CAPACITY) { + //阈值设置为最大,返回久数组 + threshold = Integer.MAX_VALUE; + return oldTab; + } + // 没超过最大值,就扩充为原来的2倍 + else if ((newCap = oldCap << 1) < MAXIMUM_CAPACITY && + oldCap >= DEFAULT_INITIAL_CAPACITY) + newThr = oldThr << 1; // double threshold + } + else if (oldThr > 0) // initial capacity was placed in threshold + //旧数组无长度,长度设置为旧的容量的大小 + newCap = oldThr; + else { // zero initial threshold signifies using defaults + newCap = DEFAULT_INITIAL_CAPACITY; + newThr = (int)(DEFAULT_LOAD_FACTOR * DEFAULT_INITIAL_CAPACITY); + } + // 计算新的resize上限 + if (newThr == 0) { + float ft = (float)newCap * loadFactor; + newThr = (newCap < MAXIMUM_CAPACITY && ft < (float)MAXIMUM_CAPACITY ? + (int)ft : Integer.MAX_VALUE); + } + threshold = newThr; + @SuppressWarnings({"rawtypes","unchecked"}) + Node[] newTab = (Node[])new Node[newCap]; + table = newTab; + if (oldTab != null) { + // 把每个bucket都移动到新的buckets中 + for (int j = 0; j < oldCap; ++j) { + Node e; + if ((e = oldTab[j]) != null) { + oldTab[j] = null; + if (e.next == null) + newTab[e.hash & (newCap - 1)] = e; + else if (e instanceof TreeNode) + ((TreeNode)e).split(this, newTab, j, oldCap); + else { // preserve order + Node loHead = null, loTail = null; + Node hiHead = null, hiTail = null; + Node next; + do { + next = e.next; + // 原索引 + if ((e.hash & oldCap) == 0) { + if (loTail == null) + loHead = e; + else + loTail.next = e; + loTail = e; + } + // 原索引+oldCap + else { + if (hiTail == null) + hiHead = e; + else + hiTail.next = e; + hiTail = e; + } + } while ((e = next) != null); + // 原索引放到bucket里 + if (loTail != null) { + loTail.next = null; + newTab[j] = loHead; + } + // 原索引+oldCap放到bucket里 + if (hiTail != null) { + hiTail.next = null; + newTab[j + oldCap] = hiHead; + } + } + } + } + } + return newTab; + } + ``` + +## 总结 + +进行扩容,会伴随着一次重新hash分配,并且会遍历hash表中所有的元素,是非常耗时的。在编写程序中,合理给定初始容量,就算适当浪费空间也要尽量避免resize。 + + + + + + + diff --git a/week_01/62/ArrayList-062.md b/week_01/62/ArrayList-062.md new file mode 100644 index 0000000..fe6779f --- /dev/null +++ b/week_01/62/ArrayList-062.md @@ -0,0 +1,20 @@ +ArrayList基于1.8源码学习 +默认容量大小是 DEFAULT_CAPACITY = 10; +1、扩容 +private void grow(int minCapacity) { + // overflow-conscious code + int oldCapacity = elementData.length; + int newCapacity = oldCapacity + (oldCapacity >> 1); + if (newCapacity - minCapacity < 0) + newCapacity = minCapacity; + if (newCapacity - MAX_ARRAY_SIZE > 0) + newCapacity = hugeCapacity(minCapacity); + // minCapacity is usually close to size, so this is a win: + elementData = Arrays.copyOf(elementData, newCapacity); + } +如果容器默认大小是10 ,当两个线程在容器有9个值得时候add数据,可能会造成数组下标越界。 +2、在HashMap中也存在modCount,起作用就是在遍历容器时,如果有add(put),remove操作会快速报错 +final void checkForComodification() { + if (modCount != expectedModCount) + throw new ConcurrentModificationException(); + } \ No newline at end of file diff --git a/week_01/62/HashMap-062.md b/week_01/62/HashMap-062.md new file mode 100644 index 0000000..b648fbc --- /dev/null +++ b/week_01/62/HashMap-062.md @@ -0,0 +1,135 @@ +基于jdk8的HashMap源码分析: +周末才有空写,写之前看了大家写的内容,有些源码写的详细的我就略过了,主要写一下自己的一些见解; +1、关于容量和负载因子,如果负载因子太小,会造成resize()频繁调用,造成新能消耗 +2、如果负载因子过大,会对容器空间的利用更加充分,但是会增加查找时间 +3、生成2的幂次方的容量,可以采用这种方法 +static final int tableSizeFor(int cap) { + int n = cap - 1; + n |= n >>> 1; + n |= n >>> 2; + n |= n >>> 4; + n |= n >>> 8; + n |= n >>> 16; + return (n < 0) ? 1 : (n >= MAXIMUM_CAPACITY) ? MAXIMUM_CAPACITY : n + 1; + } +4、添加元素 +final V putVal(int hash, K key, V value, boolean onlyIfAbsent, + boolean evict) { + Node[] tab; Node p; int n, i; + if ((tab = table) == null || (n = tab.length) == 0) //判断是否是没出初始化的容器 + n = (tab = resize()).length; //对容器初始化 + if ((p = tab[i = (n - 1) & hash]) == null) //判断队列所在的地址是否有值,没有则生成一个 node + tab[i] = newNode(hash, key, value, null); + else { + Node e; K k; + if (p.hash == hash && + ((k = p.key) == key || (key != null && key.equals(k)))) //判断key值是否相同。如果hash值不一样,则key值肯定不一样 + e = p; + else if (p instanceof TreeNode) //判断是否是红黑树结构 + e = ((TreeNode)p).putTreeVal(this, tab, hash, key, value); + else { + for (int binCount = 0; ; ++binCount) { //遍历node 下的链表,直到next没有值,在链表的最后一个添加 node + if ((e = p.next) == null) { + p.next = newNode(hash, key, value, null); + if (binCount >= TREEIFY_THRESHOLD - 1) // -1 for 1st + treeifyBin(tab, hash); + break; + } + if (e.hash == hash && + ((k = e.key) == key || (key != null && key.equals(k)))) //对于node 中可能会出现第一个key值不相同,而后面的值相同的情况 + break; + p = e; + } + } + if (e != null) { // existing mapping for key + V oldValue = e.value; + if (!onlyIfAbsent || oldValue == null) + e.value = value; + afterNodeAccess(e); + return oldValue; // 返回被替换的值 + } + } + ++modCount; + if (++size > threshold) //当前size大于下次预设值,就会触发resize() + resize(); + afterNodeInsertion(evict); + return null; + } + +5、resize()方法 +final Node[] resize() { + Node[] oldTab = table; + int oldCap = (oldTab == null) ? 0 : oldTab.length; + int oldThr = threshold; + int newCap, newThr = 0; + // 生成 新的容量大小,以及预设值 + if (oldCap > 0) { + if (oldCap >= MAXIMUM_CAPACITY) { + threshold = Integer.MAX_VALUE; + return oldTab; + } + else if ((newCap = oldCap << 1) < MAXIMUM_CAPACITY && + oldCap >= DEFAULT_INITIAL_CAPACITY) + newThr = oldThr << 1; // double threshold + } + else if (oldThr > 0) // initial capacity was placed in threshold + newCap = oldThr; + else { // zero initial threshold signifies using defaults + newCap = DEFAULT_INITIAL_CAPACITY; + newThr = (int)(DEFAULT_LOAD_FACTOR * DEFAULT_INITIAL_CAPACITY); + } + if (newThr == 0) { + float ft = (float)newCap * loadFactor; + newThr = (newCap < MAXIMUM_CAPACITY && ft < (float)MAXIMUM_CAPACITY ? + (int)ft : Integer.MAX_VALUE); + } + threshold = newThr; + @SuppressWarnings({"rawtypes","unchecked"}) + Node[] newTab = (Node[])new Node[newCap]; + table = newTab; + if (oldTab != null) { + for (int j = 0; j < oldCap; ++j) { //遍历原来的容器 + Node e; + if ((e = oldTab[j]) != null) { //对数组节点下有值得才进行处理 + oldTab[j] = null; + if (e.next == null) + newTab[e.hash & (newCap - 1)] = e; //使用新的容器(数组的下标与hash进行与运算,确定新容器(table)的数组下标) + else if (e instanceof TreeNode) + ((TreeNode)e).split(this, newTab, j, oldCap); //为红黑树处理 + else { // preserve order //对于所在数组节点有链表的情况 + Node loHead = null, loTail = null; + Node hiHead = null, hiTail = null; + Node next; + do { + next = e.next; + if ((e.hash & oldCap) == 0) { + if (loTail == null) + loHead = e; + else + loTail.next = e; + loTail = e; + } + else { + if (hiTail == null) + hiHead = e; + else + hiTail.next = e; + hiTail = e; + } + } while ((e = next) != null); + if (loTail != null) { + loTail.next = null; + newTab[j] = loHead; + } + if (hiTail != null) { + hiTail.next = null; + newTab[j + oldCap] = hiHead; + } + } + } + } + } + return newTab; + } +6、对于HashMap不是安全的,主要是发生在多线程put元素的时候,可能引起 死循环,可以参考 +https://blog.csdn.net/bjwfm2011/article/details/81076736 描述的很清楚 \ No newline at end of file diff --git a/week_01/62/LinkedList-062.md b/week_01/62/LinkedList-062.md new file mode 100644 index 0000000..3ab6879 --- /dev/null +++ b/week_01/62/LinkedList-062.md @@ -0,0 +1,14 @@ +LinkedList基于jdk1.8源码分析 +我觉得主要就是在节点上使用双向队列的方式,添加nexe,prev实现提高添加,以及删除的效率,是以空间换时间的方式。在redis中是以调表的方式, +对节点上下的引用都添加进来,实现快速查找,是以空间换时间的更加具体实现 +private static class Node { + E item; + Node next; + Node prev; + + Node(Node prev, E element, Node next) { + this.item = element; + this.next = next; + this.prev = prev; + } + } \ No newline at end of file diff --git a/week_01/65/ArrayList-065.md b/week_01/65/ArrayList-065.md new file mode 100644 index 0000000..ca01c1b --- /dev/null +++ b/week_01/65/ArrayList-065.md @@ -0,0 +1,368 @@ +# ArrayList 源码阅读笔记 + +## 一. 简介 + +ArrayList是日常编码中使用频率最高的数据结构,也是实现最简单的数据结构。它和数组一样有较高的随机访问效率,又实现了动态扩容,相当于动态数组。ArrayList是非线程安全容器。 + +## 二. 实现接口 + +ArrayList实现了List接口,是一个有序的线性集合,具有添加、删除、插入、遍历等操作。 +ArrayList实现了Cloneable接口,实现为浅拷贝。 +ArrayList实现了序列化接口Serializable,可以被序列化。 +ArrayList实现了随机访问接口RandomAccess,这是一个标记接口,实现了这个接口的集合for循环遍历效率高于iterator迭代器遍历。 + +## 三. 核心源码 + +### 1. 类属性 + +```java +/** + * 默认初始化容量 + */ +private static final int DEFAULT_CAPACITY = 10; + +/** + * 为所有空集合实例共用的空数组 + */ +private static final Object[] EMPTY_ELEMENTDATA = {}; + +/** + * 用于以无参构造方法进行初始化的空实例的共享空数组。区分它和 + * EMPTY_ELEMENTDATA 以了解添加第一个元素时需要扩容 + */ +private static final Object[] DEFAULTCAPACITY_EMPTY_ELEMENTDATA = {}; + +/** + * 存储ArrayList元素的数组缓冲区。 + * ArrayList的容量是这个数组的长度.每个空ArrayList的elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA时 + * 都会被扩容到DEFAULT_CAPACITY。 + */ +transient Object[] elementData; + +/** + * ArrayList中元素的个数 + * + */ +private int size; + +/** + * 数组最大长度。Java中的数组是由JVM构造的类,根据OOP-Klass二分模型,对象有对象头,而数组不能自己计算自己的长度,需要8字节 + * 存储长度信息,所以是Integer.MAX_VALUE - 8 + */ +private static final int MAX_ARRAY_SIZE = Integer.MAX_VALUE - 8; + +/** + * 这是一个从父类AbstractList中继承而来的属性,记录了ArrayList被结构性修改的次数 + * java.util包下的集合类都是快速失败(fail—fast)的,不能在多线程下发生并发修改(迭代过程中被修改) + */ +protected transient int modCount = 0; +``` + +### 2. 核心方法 + +#### 构造方法 + +初始容量为参数,如果大于0就初始化elementData为对应大小,如果等于0就使用EMPTY_ELEMENTDATA空数组,如果小于0抛出异常。 + +```java +public ArrayList(int initialCapacity) { + if (initialCapacity > 0) { + this.elementData = new Object[initialCapacity]; + } else if (initialCapacity == 0) { + this.elementData = EMPTY_ELEMENTDATA; + } else { + throw new IllegalArgumentException("Illegal Capacity: "+ + initialCapacity); + } +} +``` + +无参构造方法,elementData初始化为DEFAULTCAPACITY_EMPTY_ELEMENTDATA。 + +```java +public ArrayList() { + this.elementData = DEFAULTCAPACITY_EMPTY_ELEMENTDATA; +} +``` + +传入一个集合初始化,如果传入集合元素个数为0,则初始化为EMPTY_ELEMENTDATA空数组,否则使用Arrays.copyOf()拷贝到elementData。 + +```java +public ArrayList(Collection c) { + elementData = c.toArray(); + if ((size = elementData.length) != 0) { + // c.toArray might (incorrectly) not return Object[] (see 6260652) + if (elementData.getClass() != Object[].class) + elementData = Arrays.copyOf(elementData, size, Object[].class); + } else { + // replace with empty array. + this.elementData = EMPTY_ELEMENTDATA; + } +} +``` + +#### 添加元素 + +add(E e)方法将单个元素添加到ArrayList的尾部,其中涉及容量检查和扩容处理,平均时间复杂度O(1)。 + +```java +public boolean add(E e) { + ensureCapacityInternal(size + 1); // 检查是否需要扩容 + elementData[size++] = e; + return true; +} + +private void ensureCapacityInternal(int minCapacity) { + ensureExplicitCapacity(calculateCapacity(elementData, minCapacity)); +} + +private static int calculateCapacity(Object[] elementData, int minCapacity) { + /*如果elementData是由无参构造方法初始化为DEFAULTCAPACITY_EMPTY_ELEMENTDATA的,则需要扩容到DEFAULT_CAPACITY + */ + if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) { + return Math.max(DEFAULT_CAPACITY, minCapacity); + } + return minCapacity; +} + +private void ensureExplicitCapacity(int minCapacity) { + // 增加modCount,添加元素是结构性修改 + modCount++; + + // 扩容 + if (minCapacity - elementData.length > 0) + grow(minCapacity); +} + +private void grow(int minCapacity) { + int oldCapacity = elementData.length; + // 先将新容量设为旧容量的1.5倍。 >> 1 有符号右移一位,相当于除以2 + int newCapacity = oldCapacity + (oldCapacity >> 1); + if (newCapacity - minCapacity < 0) + // 如果新容量达不到需求,那容量就以minCapacity为准 + newCapacity = minCapacity; + if (newCapacity - MAX_ARRAY_SIZE > 0) + /**当传入容量参数太大,大到超过了数组的容量限定值Integer.MAX_VALUE-8却又小于整数限定值Integer.MAX_VALUE,那么新 + * 的数组容量以整数限定值Integer.MAX_VALUE为准,但是当传入的容量参数不大于数组的容量限定值时,以容量限定值 + * Integer.MAX_VALUE-8为准。 + */ + newCapacity = hugeCapacity(minCapacity); + // 拷贝元素到新扩容的数组 + elementData = Arrays.copyOf(elementData, newCapacity); +} + +private static int hugeCapacity(int minCapacity) { + if (minCapacity < 0) // 溢出报错 + throw new OutOfMemoryError(); + return (minCapacity > MAX_ARRAY_SIZE) ? + Integer.MAX_VALUE : + MAX_ARRAY_SIZE; +} +``` + +add(int index, E element)方法可以将element添加到指定的index上。 + +```java +public void add(int index, E element) { + rangeCheckForAdd(index); + + // 检查是否需要扩容 + ensureCapacityInternal(size + 1); + // 复制并调整数组内元素位置 + System.arraycopy(elementData, index, elementData, index + 1, + size - index); + elementData[index] = element; + size++; +} + +private void rangeCheckForAdd(int index) { + // 检查插入index是否合法,不合法抛出异常 + if (index > size || index < 0) + throw new IndexOutOfBoundsException(outOfBoundsMsg(index)); +} +``` + +addAll(Collection c)方法可以将一个集合中的所有元素全部有序添加到ArrayList尾部。 +addAll(int index, Collection c)方法可以将一个集合插入到ArrayList指定index上。 + +```java +public boolean addAll(Collection c) { + // 集合转数组 + Object[] a = c.toArray(); + int numNew = a.length; + // 扩容处理 + ensureCapacityInternal(size + numNew); + // 元素复制 + System.arraycopy(a, 0, elementData, size, numNew); + size += numNew; + return numNew != 0; +} + +public boolean addAll(int index, Collection c) { + // 检查index合法性 + rangeCheckForAdd(index); + + Object[] a = c.toArray(); + int numNew = a.length; + // 扩容 + ensureCapacityInternal(size + numNew); + + // 元素复制 + int numMoved = size - index; + if (numMoved > 0) + System.arraycopy(elementData, index, elementData, index + numNew, + numMoved); + + System.arraycopy(a, 0, elementData, index, numNew); + size += numNew; + return numNew != 0; +} +``` + +#### 删除元素 + +remove(int index)方法将删除指定index上的元素,并返回该元素。删除时需要进行遍历调整index,平均时间复杂度O(n)。 + +```java +public E remove(int index) { + // 检查index是否合法 + rangeCheck(index); + + // 增加modCount,删除元素也是结构性修改 + modCount++; + // 获取待删除index上的元素 + E oldValue = elementData(index); + + // 如果index不是最后一位,则将index之后的元素往前挪一位 + int numMoved = size - index - 1; + if (numMoved > 0) + System.arraycopy(elementData, index+1, elementData, index, + numMoved); + elementData[--size] = null; // 最后一个元素的位置置为null,有利于GC + + return oldValue; +} + +private void rangeCheck(int index) { + // 检查index是否越界 + if (index >= size) + throw new IndexOutOfBoundsException(outOfBoundsMsg(index)); +} + +E elementData(int index) { + return (E) elementData[index]; +} +``` + +remove(Object o)方法删除指定元素值的方法(通过equals()方法判断),平均时间复杂度O(n)。 + +```java +public boolean remove(Object o) { + if (o == null) { + // 遍历寻找index并删除null元素 + for (int index = 0; index < size; index++) + if (elementData[index] == null) { + fastRemove(index); + return true; + } + } else { + // 遍历寻找index并删除普通元素 + for (int index = 0; index < size; index++) + if (o.equals(elementData[index])) { + fastRemove(index); + return true; + } + } + return false; +} + +private void fastRemove(int index) { + // fastRemove(int index)相比remove(int index)方法,少了index越界检查。 + // 增加modCount + modCount++; + + // 如果index不是最后一位,则将index之后的元素往前挪一位 + int numMoved = size - index - 1; + if (numMoved > 0) + System.arraycopy(elementData, index+1, elementData, index, + numMoved); + elementData[--size] = null; // 最后一个元素的位置置为null,有利于GC +} +``` + +removeAll(Collection c)方法可以从ArrayList中删除一个指定集合. + +```java +public boolean removeAll(Collection c) { + // 指定集合不能为null + Objects.requireNonNull(c); + // 批量删除 + return batchRemove(c, false); +} + +/** + * 批量删除元素 + * complement为true表示删除c中不包含的元素 + * complement为false表示删除c中包含的元素 + */ +private boolean batchRemove(Collection c, boolean complement) { + final Object[] elementData = this.elementData; + // 读写分别用两个index表示 + int r = 0, w = 0; + boolean modified = false; + try { + // 遍历整个数组,根据complement把该元素放到写index的位置 + for (; r < size; r++) + if (c.contains(elementData[r]) == complement) + elementData[w++] = elementData[r]; + } finally { + // 如果contains()抛出异常,则把未读的元素都拷贝到写index之后 + if (r != size) { + System.arraycopy(elementData, r, + elementData, w, + size - r); + w += size - r; + } + if (w != size) { + // 将写index之后的元素置为null,有利于GC + for (int i = w; i < size; i++) + elementData[i] = null; + // 修改modCount + modCount += size - w; + // 新大小等于写指针的位置(因为每写一次写指针就加1,所以新大小正好等于写指针的位置) + size = w; + modified = true; + } + } + return modified; +} +``` + +#### 获取元素 + +get(int index)方法很简单,获取指定索引位置的元素,时间复杂度为O(1)。 + +```java +public E get(int index) { + // index越界检查 + rangeCheck(index); + + return elementData(index); +} +``` + +#### 设置元素 + +set(int index, E element)方法可以将指定index的元素设置为指定element,并返回旧元素,时间复杂度O(1)。 + +```java +public E set(int index, E element) { + // inde越界检查 + rangeCheck(index); + + // 获取旧元素 + E oldValue = elementData(index); + elementData[index] = element; + return oldValue; +} +``` diff --git a/week_01/65/HashMap-065.md b/week_01/65/HashMap-065.md new file mode 100644 index 0000000..7653978 --- /dev/null +++ b/week_01/65/HashMap-065.md @@ -0,0 +1,282 @@ +# HashMap源码阅读笔记 + +## 一. 简介 + +HashMap是一种key-value形式的数据结构,key和value一一对应,通过key可以快速定位到value进行操作。在HashMap中是通过key的hashcode定位value的,因此key不能为null而且不能为基本数据类型。在JDK1.7及以前,HashMap底层采用数组+链表的数据结构存储数据,这导致了一些情况下会出现性能问题,而且面对哈希洪水攻击无抵抗能力,在JDK1.8及之后,底层采用数组+链表+红黑树的数据结构存储数据来解决这些问题。HashMap是线程不安全的数据结构,且不能保证存储顺序。 + +## 二. 实现接口 + +HashMap实现了Map接口,是一个key-value数据结构,具有添加、删除、插入、遍历等操作。 +HashMap实现了Cloneable接口,实现为浅拷贝。 +HashMap实现了序列化接口Serializable,可以被序列化。 + +## 三. 核心源码 + +### 1. 底层存储结构 + +HashMap存储结构如下图所示。最基本的存储结构是数组,数组的一个元素又称作桶,当出现hash冲突时转为链表,当链表长度大于8时链表又会转为红黑树。 + +![1HashMap底层存储结构][p1] + +[p1]: + +### 2. 类属性 + +#### 内部类 + +Node静态内部类是HashMap最基本的节点,重写了equals()方法和hashcode()方法。Node数组中存放着key-value,根据key的hashcode决定存放在数组中的位置。当遇到hashcode冲突时,Node节点会组成链表以解决冲突。 + +```java +static class Node implements Map.Entry { + final int hash; + final K key; + V value; + Node next; + + Node(int hash, K key, V value, Node next) { + this.hash = hash; + this.key = key; + this.value = value; + this.next = next; + } + + public final K getKey() { return key; } + public final V getValue() { return value; } + public final String toString() { return key + "=" + value; } + + public final int hashCode() { + return Objects.hashCode(key) ^ Objects.hashCode(value); + } + + public final V setValue(V newValue) { + V oldValue = value; + value = newValue; + return oldValue; + } + + public final boolean equals(Object o) { + if (o == this) + return true; + if (o instanceof Map.Entry) { + Map.Entry e = (Map.Entry)o; + if (Objects.equals(key, e.getKey()) && + Objects.equals(value, e.getValue())) + return true; + } + return false; + } +} +``` + +TreeNode静态内部类是HashMap中的红黑树节点,当一条链表中节点大于8且HashMap中桶的数量达到64的时候,链表会转化为红黑树。 + +```java +static final class TreeNode extends LinkedHashMap.Entry +``` + +#### 属性 + +```java + +/** + * 默认初始化容量16 + */ +static final int DEFAULT_INITIAL_CAPACITY = 1 << 4; // aka 16 + +/** + * 最大容量2^30 + */ +static final int MAXIMUM_CAPACITY = 1 << 30; + +/** + * 默认装载因子0.75 + */ +static final float DEFAULT_LOAD_FACTOR = 0.75f; + +/** + * 当一个桶中的元素个数大于等于8时进行树化 + */ +static final int TREEIFY_THRESHOLD = 8; + +/** + * 当一个桶中的元素个数小于等于6时把树转化为链表 + */ +static final int UNTREEIFY_THRESHOLD = 6; + +/** + * 当桶的个数达到64的时候才进行树化 + */ +static final int MIN_TREEIFY_CAPACITY = 64; + +/** + * 桶数组 + */ +transient Node[] table; + +/** + * entrySet()方法返回的集合 + */ +transient Set> entrySet; + +/* + * 元素的数量 + */ +transient int size; + +/** + * 结构性修改次数,同ArrayList + */ +transient int modCount; + +/** + * 当桶的使用数量达到多少时进行扩容,threshold = capacity * loadFactor + */ +int threshold; + +/** + * 装载因子是表示HsahMap中元素的填满的程度。装载因子用来计算容量达到多少时才进行扩容,默认装载因子为0.75。 + */ +final float loadFactor; +``` + +### 3. 核心方法 + +#### 构造方法 + +无参构造方法 + +```java +public HashMap() { + this.loadFactor = DEFAULT_LOAD_FACTOR; // all other fields defaulted +} +``` + +传入初始容量和装载因子的构造方法,根据判断传入参数是否合法构造HashMap。 + +```java +public HashMap(int initialCapacity, float loadFactor) { + if (initialCapacity < 0) + throw new IllegalArgumentException("Illegal initial capacity: " + + initialCapacity); + if (initialCapacity > MAXIMUM_CAPACITY) + initialCapacity = MAXIMUM_CAPACITY; + if (loadFactor <= 0 || Float.isNaN(loadFactor)) + throw new IllegalArgumentException("Illegal load factor: " + + loadFactor); + this.loadFactor = loadFactor; + this.threshold = tableSizeFor(initialCapacity); +} + +// 计算当桶的数量达到多少时扩容,这里计算为初始容量向上取最接近的2的n次方 +static final int tableSizeFor(int cap) { + int n = cap - 1; + n |= n >>> 1; + n |= n >>> 2; + n |= n >>> 4; + n |= n >>> 8; + n |= n >>> 16; + return (n < 0) ? 1 : (n >= MAXIMUM_CAPACITY) ? MAXIMUM_CAPACITY : n + 1; +} +``` + +传入初始容量的构造方法,会采用默认装载因子。 + +```java +public HashMap(int initialCapacity) { + this(initialCapacity, DEFAULT_LOAD_FACTOR); +} +``` + +#### 插入元素 + +```java +public V put(K key, V value) { + return putVal(hash(key), key, value, false, true); +} + +final V putVal(int hash, K key, V value, boolean onlyIfAbsent, + boolean evict) { + Node[] tab; Node p; int n, i; + if ((tab = table) == null || (n = tab.length) == 0) + n = (tab = resize()).length; + if ((p = tab[i = (n - 1) & hash]) == null) + // 新建一个节点放在桶中 + tab[i] = newNode(hash, key, value, null); + else { + Node e; K k; + if (p.hash == hash && + ((k = p.key) == key || (key != null && key.equals(k)))) + e = p; + else if (p instanceof TreeNode) + // 如果第一个元素是树节点,则调用树节点的putTreeVal插入元素 + e = ((TreeNode)p).putTreeVal(this, tab, hash, key, value); + else { + // 遍历这个桶对应的链表,binCount用于存储链表中元素的个数 + for (int binCount = 0; ; ++binCount) { + if ((e = p.next) == null) { + p.next = newNode(hash, key, value, null); + if (binCount >= TREEIFY_THRESHOLD - 1) // -1 for 1st + treeifyBin(tab, hash); + break; + } + if (e.hash == hash && + ((k = e.key) == key || (key != null && key.equals(k)))) + break; + p = e; + } + } + if (e != null) { // existing mapping for key + V oldValue = e.value; + if (!onlyIfAbsent || oldValue == null) + e.value = value; + // 节点访问后的回调方法,在LinkedHashMap中用到 + afterNodeAccess(e); + return oldValue; + } + } + ++modCount; + if (++size > threshold) + resize(); + // 节点插入后的回调方法,在LinkedHashMap中用到 + afterNodeInsertion(evict); + return null; +} + +static final int hash(Object key) { + int h; + // 如果key为null,则hash值为0,否则调用key的hashCode()方法 + // 并让高16位与整个hash异或,这样做是为了使计算出的hash更分散 + return (key == null) ? 0 : (h = key.hashCode()) ^ (h >>> 16); +} +``` + +#### 获取元素 + +```java +public V get(Object key) { + Node e; + return (e = getNode(hash(key), key)) == null ? null : e.value; +} + +final Node getNode(int hash, Object key) { + Node[] tab; Node first, e; int n; K k; + if ((tab = table) != null && (n = tab.length) > 0 && + (first = tab[(n - 1) & hash]) != null) { + if (first.hash == hash && // always check first node + ((k = first.key) == key || (key != null && key.equals(k)))) + return first; + if ((e = first.next) != null) { + if (first instanceof TreeNode) + return ((TreeNode)first).getTreeNode(hash, key); + do { + if (e.hash == hash && + ((k = e.key) == key || (key != null && key.equals(k)))) + return e; + } while ((e = e.next) != null); + } + } + return null; +} +``` + +# 未完待续... diff --git a/week_01/65/LinkedList-065.md b/week_01/65/LinkedList-065.md new file mode 100644 index 0000000..faeed7b --- /dev/null +++ b/week_01/65/LinkedList-065.md @@ -0,0 +1,448 @@ +# LinkedList 源码阅读笔记 + +## 一. 简介 + +LinkedList是链表实现的List,内部维护了一个双向链表,插入、修改或删除元素效率较高,访问元素效率低于ArrayList。LinkedList也是一个非线程安全的集合。 + +## 二. 实现接口 + +LinkedList实现了List接口,是一个有序的线性集合,具有添加、删除、插入、遍历等操作。 +LinkedList实现了Cloneable接口,实现为浅拷贝。 +LinkedList实现了序列化接口Serializable,可以被序列化。 +LinkedList实现了双端队列接口Deque,说明LinkedList是一种具有队列和栈的性质的数据结构。 + +## 三. 核心源码 + +### 1. 类属性 + +#### 内部类 + +双链表节点Node + +```java +private static class Node { + E item; + Node next; + Node prev; + + Node(Node prev, E element, Node next) { + this.item = element; + this.next = next; + this.prev = prev; + } +} +``` + +#### 属性 + +```java +/** + * 指向双向链表第一个节点的指针 + */ +transient Node first; + +/** + * 指向双向链表最后一个节点的指针 + */ +transient Node last; + +/** + * LinkedList中元素的个数 + * + */ +transient int size; +``` + +### 2. 核心方法 + +#### 构造方法 + +无参构造方法,构造一个空List。 + +```java +public LinkedList() { +} +``` + +传入一个集合初始化,将集合中的元素添加到链表中。 + +```java +public LinkedList(Collection c) { + this(); + addAll(c); +} +``` + +#### 添加元素 + +LinkedList既可作为List,又可作为双端队列,因此它除了可以在任意位置添加元素外,还应该具有双端队列的性质,即在队列头或队列尾添加元素。 + +add(E e)方法、addLast(E e)方法、offer(E e)和offerLast(E e)都将在LinkedList尾部添加元素,平均时间复杂度O(1)。 + +```java +public boolean add(E e) { + linkLast(e); + return true; +} + +public void addLast(E e) { + linkLast(e); +} + +public boolean offer(E e) { + return add(e); +} + +public boolean offerLast(E e) { + addLast(e); + return true; +} + +void linkLast(E e) { + // 尾节点引用 + final Node l = last; + // 创建新节点,prev指向尾节点,next指向null + final Node newNode = new Node<>(l, e, null); + // 新节点成为尾节点 + last = newNode; + if (l == null) + // last节点为null说明之前LinkedList为空,则把首节点也指向新节点 + first = newNode; + else + l.next = newNode; + size++; + modCount++; +} +``` + +addFirst(E e)方法、offerFirst(E e)方法和push(E e)方法都将在LinkedList首部添加元素,平均时间复杂度O(1)。 + +```java +public void addFirst(E e) { + linkFirst(e); +} + +public boolean offerFirst(E e) { + addFirst(e); + return true; +} + +public void push(E e) { + addFirst(e); +} + +private void linkFirst(E e) { + // 首节点引用 + final Node f = first; + // 创建新节点,prev指向null,next指向首节点 + final Node newNode = new Node<>(null, e, f); + // 新节点成为首节点 + first = newNode; + if (f == null) + // first节点为null说明之前LinkedList为空,则把尾节点也指向新节点 + last = newNode; + else + f.prev = newNode; + size++; + modCount++; +} +``` + +add(int index, E element)方法可以将element添加到指定的index上,平均时间复杂度O(n)。 + +```java +public void add(int index, E element) { + // 检查index是否越界 + checkPositionIndex(index); + + if (index == size) + // 如果index == size,则添加元素到尾部 + linkLast(element); + else + // 将元素插入到index处 + linkBefore(element, node(index)); +} + +Node node(int index) { + // assert isElementIndex(index); + + // 根据index获取节点,思路为如果index < size / 2则从首节点开始往后遍历 + // 否则从尾节点往前遍历 + if (index < (size >> 1)) { + Node x = first; + for (int i = 0; i < index; i++) + x = x.next; + return x; + } else { + Node x = last; + for (int i = size - 1; i > index; i--) + x = x.prev; + return x; + } +} + +/** + * e是待插入节点,succ是待添加节点的后继节点 + */ +void linkBefore(E e, Node succ) { + // assert succ != null; + + // succ的前驱节点 + final Node pred = succ.prev; + // 创建新节点,prev指向succ的前驱节点,next指向succ + final Node newNode = new Node<>(pred, e, succ); + succ.prev = newNode; + if (pred == null) + first = newNode; + else + pred.next = newNode; + size++; + modCount++; +} +``` + +addAll(Collection c)方法可以将一个集合中的所有元素全部有序添加到LinkedList尾部。 +addAll(int index, Collection c)方法可以将一个集合插入到LinkedList指定index上。 + +```java +public boolean addAll(Collection c) { + return addAll(size, c); +} + +public boolean addAll(int index, Collection c) { + // 检查index是否越界 + checkPositionIndex(index); + + Object[] a = c.toArray(); + int numNew = a.length; + if (numNew == 0) + return false; + + Node pred, succ; + if (index == size) { + succ = null; + pred = last; + } else { + succ = node(index); + pred = succ.prev; + } + + // 迭代器遍历复制 + for (Object o : a) { + @SuppressWarnings("unchecked") E e = (E) o; + Node newNode = new Node<>(pred, e, null); + if (pred == null) + first = newNode; + else + pred.next = newNode; + pred = newNode; + } + + if (succ == null) { + last = pred; + } else { + pred.next = succ; + succ.prev = pred; + } + + size += numNew; + modCount++; + return true; +} +``` + +#### 删除元素 + +remove(int index)方法将删除指定index上的元素,并返回该元素。删除时需要进行遍历找到删除节点的位置,平均时间复杂度O(n)。 + +```java +public E remove(int index) { + // index越界检查 + checkElementIndex(index); + // 找到index位置上的元素,从链表中删除 + return unlink(node(index)); +} + +E unlink(Node x) { + // assert x != null; + final E element = x.item; + final Node next = x.next; + final Node prev = x.prev; + + if (prev == null) { + // prev == null说明待删除节点为首节点,则把待删除节点后继节点设为首节点 + first = next; + } else { + prev.next = next; + x.prev = null; + } + + if (next == null) { + // next == null说明待删除节点为尾节点,则把待删除节点前驱节点设为尾节点 + last = prev; + } else { + next.prev = prev; + x.next = null; + } + + // 有利于GC + x.item = null; + size--; + modCount++; + return element; +} +``` + +remove(Object o)方法删除指定元素值的方法(通过equals()方法判断),平均时间复杂度O(n)。 + +```java +public boolean remove(Object o) { + if (o == null) { + for (Node x = first; x != null; x = x.next) { + if (x.item == null) { + unlink(x); + return true; + } + } + } else { + for (Node x = first; x != null; x = x.next) { + if (o.equals(x.item)) { + unlink(x); + return true; + } + } + } + return false; +} +``` + +removeFirst()方法和pop()方法删除首节点元素并返回该元素,如果首节点为null则抛出异常; +poll()和pollFirst()也是删除首节点元素并返回该元素,不同的是如果首节点为null则返回null,它们的时间复杂度都为O(1)。 + +```java +public E removeFirst() { + final Node f = first; + if (f == null) + throw new NoSuchElementException(); + return unlinkFirst(f); +} + +public E poll() { + final Node f = first; + return (f == null) ? null : unlinkFirst(f); +} + +public E pollFirst() { + final Node f = first; + return (f == null) ? null : unlinkFirst(f); +} + +public E pop() { + return removeFirst(); +} + +private E unlinkFirst(Node f) { + // assert f == first && f != null; + final E element = f.item; + final Node next = f.next; + f.item = null; + f.next = null; // help GC + first = next; + if (next == null) + last = null; + else + next.prev = null; + size--; + modCount++; + return element; +} +``` + +removeLast()方法删除首节点元素并返回该元素,如果首节点为null则抛出异常; +pollLast()也是删除首节点元素并返回该元素,不同的是如果首节点为null则返回null,它们的时间复杂度都为O(1)。 + +```java +public E removeLast() { + final Node f = last; + if (f == null) + throw new NoSuchElementException(); + return unlinkLast(f); +} + +public E pollLast() { + final Node l = last; + return (l == null) ? null : unlinkLast(l); +} + +private E unlinkLast(Node f) { + // assert l == last && l != null; + final E element = l.item; + final Node prev = l.prev; + l.item = null; + l.prev = null; // help GC + last = prev; + if (prev == null) + first = null; + else + prev.next = null; + size--; + modCount++; + return element; +} +``` + +#### 获取元素 + +get(int index)方法很简单,遍历获取指定索引位置的元素,时间复杂度为O(n)。 + +```java +public E get(int index) { + checkElementIndex(index); + return node(index).item; +} +``` + +getFirst()、element()和peek()方法都返回链表的首节点,getFirst()方法和element()方法在首节点为null时将抛出异常,而peek()方法则会返回null,时间复杂度为O(1)。 + +```java +public E getFirst() { + final Node f = first; + if (f == null) + throw new NoSuchElementException(); + return f.item; +} + +public E element() { + return getFirst(); +} + +public E peek() { + final Node f = first; + return (f == null) ? null : f.item; +} +``` + +getLast()方法返回链表的尾节点,在尾节点为null时将抛出异常,时间复杂度为O(1)。 + +```java +public E getLast() { + final Node l = last; + if (l == null) + throw new NoSuchElementException(); + return l.item; +} +``` + +#### 设置元素 + +set(int index, E element)方法可以将指定index的元素设置为指定element,并返回旧元素,时间复杂度O(1)。 + +```java +public E set(int index, E element) { + checkElementIndex(index); + Node x = node(index); + E oldVal = x.item; + x.item = element; + return oldVal; +} +``` diff --git a/week_01/ArrayList.emmx b/week_01/ArrayList.emmx new file mode 100644 index 0000000..63d9ef6 Binary files /dev/null and b/week_01/ArrayList.emmx differ