起因
在Leetcode上做題寫了兩種暴力解法,但是執(zhí)行效率上不太一樣。
時間上差很遠,內(nèi)存雖然差不多但是前者擊敗30%,后者擊敗94%。這兩種解法區(qū)別是用一條ArrayList
還是兩條來存數(shù)據(jù),所以contains雖然執(zhí)行次數(shù)一樣但是檢測的長度上不一樣,而且ArrayList
的擴容次數(shù)也不一樣,所以學習一下。
contains(Object o)
直接翻(JDK8)源碼:null
和object
區(qū)分開來還是因為equals
有一方是null
的話都會導致異常. 合并一起寫的話可以用Objects.equals(obj1, obj2)
的寫法.
所以顯然暴力解法用到的contains
的原理就是樸實無華的一遍遍搜索所以時間特別長.
ArrayList擴容機制
省流: 直接看最下面的grow
函數(shù).
如果是默認的ArrayList
, 添加元素時會先計算數(shù)組長度, 如果元素個數(shù)+1大于當前數(shù)組長度+1大于elementData.length
時進行擴容,擴容后的數(shù)組大小是: oldCapacity + (oldCapacity >> 1)
,位運算右移1位表示除以2, 所以可以理解成1.5倍擴容。
涉及到的源碼:文章來源:http://www.zghlxwxcb.cn/news/detail-711403.html
// 向指定索引位置插入元素
public void add(int index, E element) {
// 檢查索引范圍
rangeCheckForAdd(index);
// 確保容量足夠
ensureCapacityInternal(size + 1); // 增加 modCount(用于支持并發(fā)修改的計數(shù)器)
// 使用 System.arraycopy 將元素后移,為新元素騰出位置, 這是跟另一個add的區(qū)別?????
System.arraycopy(elementData, index, elementData, index + 1, size - index);
elementData[index] = element; // 在指定位置插入新元素
size++; // 更新列表大小
}
// 在列表末尾添加元素
public boolean add(E e) {
// 確保容量足夠
ensureCapacityInternal(size + 1); // 增加 modCount
elementData[size++] = e; // 在列表末尾添加新元素
return true;
}
// 內(nèi)部方法:確保容量足夠
private void ensureCapacityInternal(int minCapacity) {
ensureExplicitCapacity(calculateCapacity(elementData, minCapacity));
}
// 內(nèi)部方法:計算容量
private static int calculateCapacity(Object[] elementData, int minCapacity) {
if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {
// 如果內(nèi)部數(shù)組為空,返回默認容量或所需容量中的較大者
return Math.max(DEFAULT_CAPACITY, minCapacity);
}
return minCapacity; // 否則返回所需容量
}
// 內(nèi)部方法:確保容量足夠
private void ensureExplicitCapacity(int minCapacity) {
modCount++; // 增加并發(fā)修改計數(shù)器
// 檢查容量是否足夠,如果不夠則擴展
if (minCapacity - elementData.length > 0)
grow(minCapacity);
}
// 內(nèi)部方法:擴展容量
private void grow(int minCapacity) {
// 溢出安全的代碼
int oldCapacity = elementData.length;
int newCapacity = oldCapacity + (oldCapacity >> 1); // 新容量通常為舊容量的1.5倍
if (newCapacity - minCapacity < 0)
newCapacity = minCapacity; // 如果新容量小于所需容量,使用所需容量
if (newCapacity - MAX_ARRAY_SIZE > 0)
newCapacity = hugeCapacity(minCapacity); // 處理可能的巨大容量情況
// 使用 Arrays.copyOf 擴展數(shù)組容量
elementData = Arrays.copyOf(elementData, newCapacity);
}
實際上Array.copyof
底層調(diào)用的還是System.arraycopy
.文章來源地址http://www.zghlxwxcb.cn/news/detail-711403.html
到了這里,關(guān)于學習一下Java的ArrayList和contains函數(shù)和擴容機制的文章就介紹完了。如果您還想了解更多內(nèi)容,請在右上角搜索TOY模板網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章,希望大家以后多多支持TOY模板網(wǎng)!