前期自己對(duì)于源碼解析,以為看過既可以,然后就是類似翻譯的動(dòng)作而已。后期發(fā)現(xiàn)這種形式并沒有什么好的。所以需要重新思考。
看源碼前的準(zhǔn)備。
1.ArrayList是什么
首先根據(jù)如下圖:

1.最頂級(jí)接口Iterable,這個(gè)是迭代器模式的頂級(jí)接口,忽略
2.Collection接口,這個(gè)就是集合類的頂級(jí)接口了
3.List接口,列表類頂級(jí)接口,于set集合區(qū)分
4.RandomAccess接口,是一個(gè)標(biāo)記接口。表明當(dāng)前類可以進(jìn)行快速隨機(jī)訪問。忽略
5.Cloneable接口,標(biāo)記接口。表明深拷貝。忽略
6.其他的序列化接口,和集合的抽象實(shí)現(xiàn),列表的抽象實(shí)現(xiàn)。
所以從ArrayList的類圖可以看出,ArrayList就是一個(gè)集合類。且是一個(gè)列表集合,即支持可重復(fù)的集合元素。
2.ArrayList能干什么
通過前面的分析,知道ArrayList是一個(gè)列表集合的具體實(shí)現(xiàn)。那么列表集合能干什么呢。最基礎(chǔ)的應(yīng)用就是存儲(chǔ)。將數(shù)據(jù)存儲(chǔ)到內(nèi)存中。且存儲(chǔ)的內(nèi)容是按照存入的先后順序進(jìn)行存儲(chǔ)的。
3.分析準(zhǔn)備
在進(jìn)行ArrayList分析之前,ArrayList是什么,能干什么我們已經(jīng)很清楚了。那么首先我們自己先分析下,如果自己去寫一個(gè)集合都需要干什么呢。
3.1類比
這種分析一般都是需要對(duì)照到生活中的例子進(jìn)行類比分析,然后進(jìn)行代碼實(shí)現(xiàn),才能更好的理解類。首先集合類似于我們生活中的倉庫。
3.2操作
既然是倉庫那么就需要存儲(chǔ)貨物。而存儲(chǔ)貨物的時(shí)候最基本的操作。
1.入庫
2.出庫
3.清點(diǎn)貨物總數(shù)
4.查看貨物
3.3自己實(shí)現(xiàn)
根據(jù)自己的思路實(shí)現(xiàn)一個(gè)ArrayList
package com.jiyx.test.collect;
/**
*
* auther: jiyx
* date: 2019-05-02
*/
public class MyArrayList {
private Object[] items = new Object[64];
private int currIndex;
/**
* 入庫
* @param item
*/
public void add(Object item) {
rangeCheckOut(currIndex);
items[currIndex++] = item;
}
/**
* 查看,不出庫
* @param index
* @return
*/
public Object get(int index) {
rangeCheckOut(index);
return items[index];
}
/**
* 出庫
* @param index
* @return
*/
public Object remove(int index) {
rangeCheckOut(index);
Object value = items[index];
if (items.length - index - 1 > 0) {
System.arraycopy(items, index + 1, items, index, items.length - index - 1);
}
currIndex--;
return value;
}
/**
* 當(dāng)前存儲(chǔ)的元素個(gè)數(shù)
* @return
*/
public int size() {
return currIndex;
}
/**
* 角標(biāo)越界校驗(yàn)
* @param currIndex
*/
private void rangeCheckOut(int currIndex) {
if (currIndex > items.length - 1) {
throw new ArrayIndexOutOfBoundsException();
}
}
public static void main(String[] args) {
MyArrayList list = new MyArrayList();
list.add(1);
list.add("aaa");
list.add(list);
for (int i = 0; i < list.size(); i++) {
System.out.println(list.get(i));
}
list.remove(1);
list.add("bbb");
for (int i = 0; i < list.size(); i++) {
System.out.println(list.get(i));
}
}
}
4.分析
好了進(jìn)入正題,進(jìn)行ArrayList源碼淺析。這里ArrayList無非也是上面的幾個(gè)操作,不過可能比我寫的更加完善,功能更多而已。
那么同樣的,還是按照基本功能,外加一個(gè)初始化進(jìn)行分析。
4.1初始化
因?yàn)閭}庫是沒有的,所以我們需要自己手動(dòng)創(chuàng)建一個(gè)倉庫的。那么new的過程就是創(chuàng)建這個(gè)倉庫的過程。
初始化分為三種,無參構(gòu)造、傳入初始容量構(gòu)造器、傳入老的集合的構(gòu)造器。其實(shí)初始化的過程很簡(jiǎn)單。可以自行查看。不過這里有兩個(gè)變量需要注意。EMPTY_ELEMENTDATA和DEFAULTCAPACITY_EMPTY_ELEMENTDATA。因?yàn)樵趧?chuàng)建的過程無參構(gòu)造時(shí)需要用到后者。其他兩個(gè)構(gòu)造都是用到了前者。這兩個(gè)雖然值和存在的意義都一樣。區(qū)別是,在第一個(gè)元素添加的時(shí)候。前者的第一次擴(kuò)容為1。而后者的第一次擴(kuò)容為10。
4.1入庫
Arraylist的入庫入口為add方法。重載了兩個(gè)方法。
public boolean add(E e) {
// 判斷容量是否足夠,如果不足進(jìn)行擴(kuò)容
ensureCapacityInternal(size + 1);
// 添加元素
elementData[size++] = e;
return true;
}
public void add(int index, E element) {
rangeCheckForAdd(index);
// 判斷容量是否足夠,如果不足進(jìn)行擴(kuò)容
ensureCapacityInternal(size + 1);
// 騰出需要插入元素的位置,也就是將元素后移一位
System.arraycopy(elementData, index, elementData, index + 1,
size - index);
// 添加元素
elementData[index] = element;
size++;
}
而且這里面必須介紹下方法ensureCapacityInternal及其設(shè)計(jì)到的方法,都不是很復(fù)雜。如下:
private void ensureCapacityInternal(int minCapacity) {
ensureExplicitCapacity(calculateCapacity(elementData, minCapacity));
}
/**
* 計(jì)算最小容量
* @param elementData
* @param minCapacity
* @return
*/
private static int calculateCapacity(Object[] elementData, int minCapacity) {
if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {
return Math.max(DEFAULT_CAPACITY, minCapacity);
}
return minCapacity;
}
/**
* 根據(jù)最小容量確認(rèn)是否需要進(jìn)行擴(kuò)容
* @param minCapacity
*/
private void ensureExplicitCapacity(int minCapacity) {
modCount++;
// overflow-conscious code
if (minCapacity - elementData.length > 0)
grow(minCapacity);
}
/**
* 具體的擴(kuò)容操作
* @param 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);
// minCapacity is usually close to size, so this is a win:
elementData = Arrays.copyOf(elementData, newCapacity);
}
4.2出庫
出庫的動(dòng)作是由remove完成的。這個(gè)也比較簡(jiǎn)單了。但是呢,重載了兩個(gè),因?yàn)樵趶膫}庫中移除物品的時(shí)候??赡芨鶕?jù)編號(hào),也可能根據(jù)某個(gè)物品。
public E remove(int index) {
// 判斷角標(biāo)是否越界
rangeCheck(index);
modCount++;
E oldValue = elementData(index);
// 這塊判斷是否需要進(jìn)行數(shù)據(jù)遷移,如果移出的位置在中間,那么需要將后面的物品前移一位
int numMoved = size - index - 1;
if (numMoved > 0)
System.arraycopy(elementData, index+1, elementData, index,
numMoved);
// 將最后一位設(shè)置為null
elementData[--size] = null;
return oldValue;
}
/**
* 移出,不返回對(duì)象。因?yàn)槲覀冎酪瞥龅氖悄膫€(gè)對(duì)象
* @param o
* @return
*/
public boolean remove(Object o) {
if (o == null) {
// 移除null值
for (int index = 0; index < size; index++)
if (elementData[index] == null) {
// 真正的移出數(shù)據(jù)
fastRemove(index);
return true;
}
} else {
for (int index = 0; index < size; index++)
// 對(duì)象的移出,是根據(jù)equals進(jìn)行判斷的,所以對(duì)象要重寫這個(gè)方法
if (o.equals(elementData[index])) {
fastRemove(index);
return true;
}
}
return false;
}
/**
* 相對(duì)于remove(int index)方法,少了角標(biāo)校驗(yàn),也沒有返回值
* 因?yàn)檎{(diào)用方法的也就remove(Object o),而他是在循環(huán)遍歷中調(diào)用的,所以這里不做角標(biāo)校驗(yàn)
* @param index
*/
private void fastRemove(int index) {
// 修改次數(shù)加一
modCount++;
// 是否需要進(jìn)行數(shù)據(jù)遷移
int numMoved = size - index - 1;
if (numMoved > 0)
System.arraycopy(elementData, index+1, elementData, index,
numMoved);
elementData[--size] = null;
}
4.3清點(diǎn)貨物總數(shù)
因?yàn)橐话阄覀兊膫}庫管理時(shí),在物品入庫的時(shí)候,會(huì)做一個(gè)記錄。所以直接查看記錄就知道有多少物品了。
public int size() {
return size;
}
4.4查看貨物
get方法,比較簡(jiǎn)單
public E get(int index) {
rangeCheck(index);
return elementData(index);
}
E elementData(int index) {
return (E) elementData[index];
}
4.5其他方法
當(dāng)然了,ArrayList中還有很多實(shí)用的方法。如
/**
* 是否包含某對(duì)象
* @param o
* @return
*/
public boolean contains(Object o) {
return indexOf(o) >= 0;
}
/**
* 查看對(duì)象在倉庫arrayList中的位置
* @param o
* @return
*/
public int indexOf(Object o) {
if (o == null) {
for (int i = 0; i < size; i++)
if (elementData[i]==null)
return i;
} else {
for (int i = 0; i < size; i++)
if (o.equals(elementData[i]))
return i;
}
return -1;
}
/**
* 查找最近入庫的相同的對(duì)象
* @param o
* @return
*/
public int lastIndexOf(Object o) {
if (o == null) {
for (int i = size-1; i >= 0; i--)
if (elementData[i]==null)
return i;
} else {
for (int i = size-1; i >= 0; i--)
if (o.equals(elementData[i]))
return i;
}
return -1;
}
/**
* 批量的入庫
* @param c
* @return
*/
public boolean addAll(Collection<? extends E> c) {
Object[] a = c.toArray();
int numNew = a.length;
ensureCapacityInternal(size + numNew);
System.arraycopy(a, 0, elementData, size, numNew);
size += numNew;
return numNew != 0;
}
/**
* 從指定位置開始批量入庫
* @param index
* @param c
* @return
*/
public boolean addAll(int index, Collection<? extends E> c) {
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;
}
5.可修改的問題
1.方法中會(huì)出現(xiàn)a-b>0這種代碼。感覺沒有a>b來的清晰明了
2.代碼中存在很多重復(fù)代碼。如remove(Object o)就是修改為如下
public boolean remove(Object o) {
for (int index = 0; index < size; index++) {
if ((o == null && elementData[index] == null)
|| (o != null && o.equals(elementData[index]))) {
fastRemove(index);
return true;
}
}
return false;
}