Android4.4之后推出的ArrayMap集合详解

通常我们在使用key-value存储数据时,随手就会打出HashMap的代码,当数据量较小时,还可以,当数量比较多的时候,如果是PC机上,也还说得过去,但是如果使用设备是手机等移动设备,这是就要慎重了。因为手机的内存非常宝贵,不像PC那样不计后果的使用,内存使用不当很容易就会引起OOM的问题。那Android开发团队,也为我们找到了HashMap的替代品ArrayMap。

官方对ArrayMap也有说明:它不是一个适应大数据的数据结构,相比传统的HashMap速度要慢,因为查找方法是二分法,并且当你删除或者添加数据时,会对空间重新调整,在使用大量数据时,效率并不明显,低于50%。

所以ArrayMap是牺牲了时间换区空间。在写手机app时,适时的使用ArrayMap,会给内存使用带来可观的提升。

那HashMap和ArrayMap到底不同在哪呢,个人总结有以下方面:

1、存储方式不同。

HashMap内部有一个HashMapEntry[]对象,每一个键值对都存储在这个对象里,当使用put方法添加键值对时,就会new一个HashMapEntry对象

[java]view plaincopy

@OverridepublicV put(K key, V value) {

if(key ==null) {

returnputValueForNullKey(value);

}

inthash = secondaryHash(key);

HashMapEntry[] tab = table;

intindex = hash & (tab.length -1);

//先查找有没有对应的key值,如果有,就改写value,并返回改写前的value值:oldValue

for(HashMapEntry e = tab[index]; e !=null; e = e.next) {

if(e.hash == hash && key.equals(e.key)) {

preModify(e);

V oldValue = e.value;

e.value = value;

returnoldValue;

}

}

// No entry for (non-null) key is present; create one

modCount++;

if(size++ > threshold) {

//扩容,双倍

tab = doubleCapacity();

index = hash & (tab.length -1);

}

addNewEntry(key, value, hash, index);

returnnull;

}

//创建对象存储键值对

voidaddNewEntry(K key, V value,inthash,intindex) {

table[index] =newHashMapEntry(key, value, hash, table[index]);

}

ArrayMap的存储中没有Entry这个东西,他是由两个数组来维护的

[java]view plaincopy

int[] mHashes;

Object[] mArray;

mHashes数组中保存的是每一项的HashCode值,mArray中就是键值对,每两个元素代表一个键值对,前面保存key,后面的保存value,我们看看下面代码的结果

[java]view plaincopy

arraymap =newHashMap();

a.put("a","a_value");

a.put("b","b_value");

执行上面代码后,arraymap中的存储是这样的

是不是能清楚地看到ArrayMap的存储了,这种存储在put代码中如下

[java]view plaincopy

mHashes[index] = hash;

mArray[index<<1] = key;

mArray[(index<<1)+1] = value;

2、添加数据时扩容时的处理不一样

先来看看HashMap

[java]view plaincopy

if(size++ > threshold) {

tab = doubleCapacity();

index = hash & (tab.length -1);

}

doubleCapacity进行双倍扩容,它的代码中有这么一句话

[java]view plaincopy

HashMapEntry[] newTable = makeTable(newCapacity);

最终,这个newTable将作为扩容后的新对象返回,那么makeTable做了什么呢,如下:

[java]view plaincopy

privateHashMapEntry[] makeTable(intnewCapacity) {

@SuppressWarnings("unchecked") HashMapEntry[] newTable

= (HashMapEntry[])newHashMapEntry[newCapacity];

table = newTable;

threshold = (newCapacity >>1) + (newCapacity >>2);// 3/4 capacity

returnnewTable;

}

我们清楚地看到,这里进行了new操作,重新创建对象,开销很大。

那么ArrayMap呢,看看吧

[java]view plaincopy

//如果容量不够

ize >= mHashes.length) {

finalintn = mSize >= (BASE_SIZE*2) ? (mSize+(mSize>>1))

: (mSize >= BASE_SIZE ? (BASE_SIZE*2) : BASE_SIZE);

if(DEBUG) Log.d(TAG,"put: grow from "+ mHashes.length +" to "+ n);

finalint[] ohashes = mHashes;

finalObject[] oarray = mArray;

//分配数组

allocArrays(n);

if(mHashes.length >0) {

if(DEBUG) Log.d(TAG,"put: copy 0-"+ mSize +" to 0");

//特别注意这,是copy,而不是new,效率提升

System.arraycopy(ohashes,0, mHashes,0, ohashes.length);

System.arraycopy(oarray,0, mArray,0, oarray.length);

}

//释放无用空间,收缩数组

freeArrays(ohashes, oarray, mSize);

}

ArrayMap用的是copy数据,所以效率相对要高。

3、ArrayMap提供了数组收缩的功能,在clear或remove后,会重新收缩数组,是否空间

4、ArrayMap采用二分法查找(见 android.support.v4.util.ContainerHelpers中的binarySearch方法)

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 211,884评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,347评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,435评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,509评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,611评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,837评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,987评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,730评论 0 267
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,194评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,525评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,664评论 1 340
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,334评论 4 330
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,944评论 3 313
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,764评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,997评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,389评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,554评论 2 349

推荐阅读更多精彩内容

  • 一、基本数据类型 注释 单行注释:// 区域注释:/* */ 文档注释:/** */ 数值 对于byte类型而言...
    龙猫小爷阅读 4,257评论 0 16
  • 背景 一年多以前我在知乎上答了有关LeetCode的问题, 分享了一些自己做题目的经验。 张土汪:刷leetcod...
    土汪阅读 12,738评论 0 33
  • 宣堡,又称宣家堡;在宣堡郊区,耸立着一个巍峨高大的牌匾刻着“宣家堡”;我的那个他用地道的方言告诉我,叫“宣嘎堡”。...
    三末阅读 458评论 0 1
  • 这本书其实是让我有些失望的,也有可能是期望过高了些。 其实在看《挪威的森林》时,尽管故事讲述的是主角渡边纠缠在情绪...
    七筒妹妹阅读 749评论 0 0
  • 生活是件细碎的事 在千千细碎中是否有一片是我对你的牵挂 是否有一种羁绊在你碌碌中沉寂 又在某个午夜梦回时给你清醒的...
    南方的茶阅读 165评论 0 0