1、SparseArray內(nèi)部使用雙數(shù)組,分別存儲Key和Value,Key是int[],用于查找Value對應(yīng)的Index,來定位數(shù)據(jù)在Value中的位置。
2、使用二分查找來定位Key數(shù)組中對應(yīng)值的位置,所以Key數(shù)組是有序的。
3、使用數(shù)組就面臨刪除數(shù)據(jù)時(shí)數(shù)據(jù)遷移的問題,所以引入了DELETE標(biāo)記。
## 雙數(shù)組

mValue數(shù)組和mKey數(shù)組的關(guān)系是一一對應(yīng)的,通過Key的值,可以定位出該Key在mKey數(shù)組中的index,進(jìn)而可以操作MValue數(shù)組中對應(yīng)位置的數(shù)據(jù)。
## 二分查找
```java
public E get(int key, E valueIfKeyNotFound) {
int i = ContainerHelpers.binarySearch(mKeys, mSize, key);
if (i < 0 || mValues[i] == DELETED) {
return valueIfKeyNotFound;
} else {
return (E) mValues[i];
}
}
```
SparseArray源碼中,所有和Key有關(guān)的操作,第一步都是先通過二分查找,定位出該Key的index,再進(jìn)行后續(xù)處理。
二分查找的前提條件,就是必須是針對有序并且支持下標(biāo)隨機(jī)訪問的數(shù)據(jù)結(jié)構(gòu),所以它在執(zhí)行插入操作的時(shí)候,必須保證 mKey 數(shù)據(jù)中的數(shù)據(jù)有序。
SparseArray的插入代碼如下:
```java
public void put(int key, E value) {
int i = ContainerHelpers.binarySearch(mKeys, mSize, key);
if (i >= 0) {
mValues[i] = value;
} else {
i = ~i;
if (i < mSize && mValues[i] == DELETED) {
mKeys[i] = key;
mValues[i] = value;
return;
}
if (mGarbage && mSize >= mKeys.length) {
gc();
// Search again because indices may have changed.
i = ~ContainerHelpers.binarySearch(mKeys, mSize, key);
}
mKeys = GrowingArrayUtils.insert(mKeys, mSize, i, key);
mValues = GrowingArrayUtils.insert(mValues, mSize, i, value);
mSize++;
}
}
```
同樣會(huì)使用二分定位查找Key的index,GrowingArrayUtils的insert方法會(huì)完成兩個(gè)任務(wù):
1、將數(shù)據(jù)插入到指定數(shù)組對應(yīng)的位置上。
2、如果發(fā)現(xiàn)數(shù)組空間不夠,就生成一個(gè)更大的新數(shù)組,將數(shù)組通過復(fù)制的方式,動(dòng)態(tài)擴(kuò)容后搬到新數(shù)組中,并返回新數(shù)組。
## DELETE標(biāo)記

對數(shù)組的刪除,如果不做數(shù)據(jù)遷移,數(shù)組中必然存在數(shù)據(jù)空洞,SparseArray引入DELETE標(biāo)記,來減少刪除數(shù)據(jù)時(shí)對數(shù)據(jù)的搬運(yùn)次數(shù)。
在插入時(shí),如遇到DELETE標(biāo)識,標(biāo)識當(dāng)前數(shù)據(jù)已經(jīng)刪除掉了,直接進(jìn)行替換,減少一次插入數(shù)據(jù)帶來的數(shù)據(jù)搬運(yùn)。

DELETE標(biāo)識是在mValue數(shù)組中存儲的,mKey中仍然存儲著它上一次對應(yīng)數(shù)據(jù)的Key值。在一些必要條件下,會(huì)觸發(fā)gc()邏輯,來清理數(shù)組中的DELETE標(biāo)識。在gc()方法中,用了一個(gè)布爾類型的mGarbage屬性,來記錄當(dāng)前 mValue 中,是否存在 DELETE 標(biāo)識,這是判定是否需要 GC 的依據(jù)。SparseArray的**所有和 size 相關(guān)的操作**以及**和數(shù)組擴(kuò)容相關(guān)的操作**時(shí)需要進(jìn)行g(shù)c操作。
## 參考文檔
[https://juejin.im/post/5da1481e6fb9a04de96e8b72](https://juejin.im/post/5da1481e6fb9a04de96e8b72)
- 導(dǎo)讀
- Java知識
- Java基本程序設(shè)計(jì)結(jié)構(gòu)
- 【基礎(chǔ)知識】Java基礎(chǔ)
- 【源碼分析】Okio
- 【源碼分析】深入理解i++和++i
- 【專題分析】JVM與GC
- 【面試清單】Java基本程序設(shè)計(jì)結(jié)構(gòu)
- 對象與類
- 【基礎(chǔ)知識】對象與類
- 【專題分析】Java類加載過程
- 【面試清單】對象與類
- 泛型
- 【基礎(chǔ)知識】泛型
- 【面試清單】泛型
- 集合
- 【基礎(chǔ)知識】集合
- 【源碼分析】SparseArray
- 【面試清單】集合
- 多線程
- 【基礎(chǔ)知識】多線程
- 【源碼分析】ThreadPoolExecutor源碼分析
- 【專題分析】volatile關(guān)鍵字
- 【面試清單】多線程
- Java新特性
- 【專題分析】Lambda表達(dá)式
- 【專題分析】注解
- 【面試清單】Java新特性
- Effective Java筆記
- Android知識
- Activity
- 【基礎(chǔ)知識】Activity
- 【專題分析】運(yùn)行時(shí)權(quán)限
- 【專題分析】使用Intent打開三方應(yīng)用
- 【源碼分析】Activity的工作過程
- 【面試清單】Activity
- 架構(gòu)組件
- 【專題分析】MVC、MVP與MVVM
- 【專題分析】數(shù)據(jù)綁定
- 【面試清單】架構(gòu)組件
- 界面
- 【專題分析】自定義View
- 【專題分析】ImageView的ScaleType屬性
- 【專題分析】ConstraintLayout 使用
- 【專題分析】搞懂點(diǎn)九圖
- 【專題分析】Adapter
- 【源碼分析】LayoutInflater
- 【源碼分析】ViewStub
- 【源碼分析】View三大流程
- 【源碼分析】觸摸事件分發(fā)機(jī)制
- 【源碼分析】按鍵事件分發(fā)機(jī)制
- 【源碼分析】Android窗口機(jī)制
- 【面試清單】界面
- 動(dòng)畫和過渡
- 【基礎(chǔ)知識】動(dòng)畫和過渡
- 【面試清單】動(dòng)畫和過渡
- 圖片和圖形
- 【專題分析】圖片加載
- 【面試清單】圖片和圖形
- 后臺任務(wù)
- 應(yīng)用數(shù)據(jù)和文件
- 基于網(wǎng)絡(luò)的內(nèi)容
- 多線程與多進(jìn)程
- 【基礎(chǔ)知識】多線程與多進(jìn)程
- 【源碼分析】Handler
- 【源碼分析】AsyncTask
- 【專題分析】Service
- 【源碼分析】Parcelable
- 【專題分析】Binder
- 【源碼分析】Messenger
- 【面試清單】多線程與多進(jìn)程
- 應(yīng)用優(yōu)化
- 【專題分析】布局優(yōu)化
- 【專題分析】繪制優(yōu)化
- 【專題分析】內(nèi)存優(yōu)化
- 【專題分析】啟動(dòng)優(yōu)化
- 【專題分析】電池優(yōu)化
- 【專題分析】包大小優(yōu)化
- 【面試清單】應(yīng)用優(yōu)化
- Android新特性
- 【專題分析】狀態(tài)欄、ActionBar和導(dǎo)航欄
- 【專題分析】應(yīng)用圖標(biāo)、通知欄適配
- 【專題分析】Android新版本重要變更
- 【專題分析】唯一標(biāo)識符的最佳做法
- 開源庫源碼分析
- 【源碼分析】BaseRecyclerViewAdapterHelper
- 【源碼分析】ButterKnife
- 【源碼分析】Dagger2
- 【源碼分析】EventBus3(一)
- 【源碼分析】EventBus3(二)
- 【源碼分析】Glide
- 【源碼分析】OkHttp
- 【源碼分析】Retrofit
- 其他知識
- Flutter
- 原生開發(fā)與跨平臺開發(fā)
- 整體歸納
- 狀態(tài)及狀態(tài)管理
- 零碎知識點(diǎn)
- 添加Flutter到現(xiàn)有應(yīng)用
- Git知識
- Git命令
- .gitignore文件
- 設(shè)計(jì)模式
- 創(chuàng)建型模式
- 結(jié)構(gòu)型模式
- 行為型模式
- RxJava
- 基礎(chǔ)
- Linux知識
- 環(huán)境變量
- Linux命令
- ADB命令
- 算法
- 常見數(shù)據(jù)結(jié)構(gòu)及實(shí)現(xiàn)
- 數(shù)組
- 排序算法
- 鏈表
- 二叉樹
- 棧和隊(duì)列
- 算法時(shí)間復(fù)雜度
- 常見算法思想
- 其他技術(shù)
- 正則表達(dá)式
- 編碼格式
- HTTP與HTTPS
- 【面試清單】其他知識
- 開發(fā)歸納
- Android零碎問題
- 其他零碎問題
- 開發(fā)思路
