Sparse array系列

1、特點:

時間換空間, 操作容器花費更多時間, 但是會減少內存;

這篇博客重點分析:

1. SparseArray;
2. SparseIntArray;
3. ArrayMap;

2、以SparseIntArray為例:

1、構造函數:
private int[] mKeys;
    private int[] mValues;
    private int mSize;
    public SparseIntArray() {
        this(10);
    }
public SparseIntArray(int initialCapacity) {
    if (initialCapacity == 0) {
        mKeys = EmptyArray.INT;
        mValues = EmptyArray.INT;
    } else {
        mKeys = ArrayUtils.newUnpaddedIntArray(initialCapacity);
        mValues = new int[mKeys.length];
    }
    mSize = 0;
}
libcore/luni/src/main/java/libcore/util/EmptyArray.java--->
public final class EmptyArray {
    private EmptyArray() {}

    public static final boolean[] BOOLEAN = new boolean[0];
    public static final byte[] BYTE = new byte[0];
    public static final char[] CHAR = new char[0];
    public static final double[] DOUBLE = new double[0];
    public static final float[] FLOAT = new float[0];
    public static final int[] INT = new int[0];
    public static final long[] LONG = new long[0];

    public static final Class<?>[] CLASS = new Class[0];
    public static final Object[] OBJECT = new Object[0];
    public static final String[] STRING = new String[0];
    public static final Throwable[] THROWABLE = new Throwable[0];
}

上面代碼主要做了一件事:

  • 1、創建兩個數組, 分別存儲key和value;
2、put():
class SparseIntArray--->
public void put(int key, int value) {
    int i = ContainerHelpers.binarySearch(mKeys, mSize, key);
    if (i >= 0) {
        mValues[i] = value;
    } else {
        i = ~i;
        mKeys = GrowingArrayUtils.insert(mKeys, mSize, i, key);
        mValues = GrowingArrayUtils.insert(mValues, mSize, i, value);
        mSize++;
    }
}

class ContainerHelpers--->
static int binarySearch(int[] array, int size, int value) {
    int lo = 0;
    int hi = size - 1;
    while (lo <= hi) {
        final int mid = (lo + hi) >>> 1;
        final int midVal = array[mid];
        if (midVal < value) {
            lo = mid + 1;
        } else if (midVal > value) {
            hi = mid - 1;
        } else {
            return mid;  
        }
    }
    return ~lo;  
}

class GrowingArrayUtils --->
public static int[] insert(int[] array, int currentSize, int index, int element) {
    assert currentSize <= array.length;
    if (currentSize + 1 <= array.length) {
        System.arraycopy(array, index, array, index + 1, currentSize - index);
        array[index] = element;
        return array;
    }
    int[] newArray = new int[growSize(currentSize)];
    System.arraycopy(array, 0, newArray, 0, index);
    newArray[index] = element;
    System.arraycopy(array, index, newArray, index + 1, array.length - index);
    return newArray;
}
public static int growSize(int currentSize) {
    return currentSize <= 4 ? 8 : currentSize * 2;
}

put主要做了下面幾件事:

  • 1、插入操作先通過二分查找, 找出key. 二分查找時間復雜度為O(logN); 而HashMap內部維護的是一個數組, 查找時間復雜度為O(1); 所以就插入操作來說, HashMap會更快一些;
  • 2、這個二分查找進行的挺巧妙的, 插入失敗返回~lo; 對返回值再繼續進行取反操作;
  • 3、然后依次將key和value插入到數組中去, 我們看看他的插入操作;
  • 4、假設我們插入操作一直是亂序的, 并且很頻繁, 會出現什么問題? 二分查找頻繁執行. 每次都是O(logN), 如果數據量過大, 與HashMap O(1)比起來時間差距就會越來越明顯;
  • 5、System.arraycopy(array, index, array, index + 1, currentSize - index)與array[index] = element;這兩步操作就是將我們要插入的元素插入到指定位置. 畫個圖
調用arraycopy以后.png
2、get():
public int get(int key, int valueIfKeyNotFound) {
      int i = ContainerHelpers.binarySearch(mKeys, mSize, key);
      if (i < 0) {
          return valueIfKeyNotFound;
      } else {
          return mValues[i];
      }
}

**查找就很簡單了, 因為在put方法執行時的一系列操作可以保證數組是有序的, 所以查找的時間復雜度也是O(logN). ** 所以如果涉及到大數據量或者查找特別頻繁的操作, SparseArray系列并不適用;

3、delete():
public void delete(int key) {
    int i = ContainerHelpers.binarySearch(mKeys, mSize, key);
    if (i >= 0) {
        removeAt(i);
    }
}
public void removeAt(int index) {
    System.arraycopy(mKeys, index + 1, mKeys, index, mSize - (index + 1));
    System.arraycopy(mValues, index + 1, mValues, index, mSize - (index + 1));
    mSize--;
}

刪除操作也是挺簡單的, 時間復雜度也是為O(logN). 因為他進行的數組元素的整體拷貝. 與get方法一樣, 當刪除操作特別頻繁或者數據量很大時, 也是沒必要使用的;

綜上所述:
當涉及到數據量不是特別大時, 官方給出的建議是千以下, 并且操作不是特別頻繁時, 使用SparseArray系列. 因為它可以節省內存空間, APP與Windows不一樣, 虛擬機為每個APP分配的內存空間很有限, SparseArray主要從以下幾個方面節省內存空間:

  • 1、沒有自動裝箱的操作;
  • 2、沒有Entry對象;及其Hash映射關系;

3、ArrayMap:

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 229,908評論 6 541
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 99,324評論 3 429
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 178,018評論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,675評論 1 317
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,417評論 6 412
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,783評論 1 329
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,779評論 3 446
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,960評論 0 290
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,522評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 41,267評論 3 358
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,471評論 1 374
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 39,009評論 5 363
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,698評論 3 348
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,099評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,386評論 1 294
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,204評論 3 398
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,436評論 2 378

推薦閱讀更多精彩內容

  • Java8張圖 11、字符串不變性 12、equals()方法、hashCode()方法的區別 13、...
    Miley_MOJIE閱讀 3,722評論 0 11
  • Android開發者都知道Lint在我們使用HashMap的時候會給出警告——使用SparseArray會優化內存...
    扈扈哈嘿閱讀 17,831評論 2 21
  • 都知道這么一句話,沒什么也不能沒錢,有什么也不能有病。 沒錢的日子,真的很苦; 有病的日子,真的很難受。 如果,生...
    煩人的昵稱閱讀 164評論 0 0
  • 阿凡提真棒
    阿提蘭閱讀 218評論 0 0
  • c語言中包含char、int、float、double等基本數據類型,本節主要研究一下這些基本數據類型的特點及存儲...
    labi3285閱讀 164評論 0 0