Android知识点 ArrayMap SparseArray

ArrayMap SparseArray

问题:ArrayMap SparseArray的数据结构是怎么样的?

双数组结构。ArrayMap第一个数组元素是key的hashValue,对应第二个数组的一对key-value。通过二分查找进行插入。 SparseArray第一个数组是int类型的key,第二个数组元素是value

为了更进一步优化key是int类型的Map,Android再次提供效率更高的数据结构SparseArray,可避免自动装箱过程。对于key为其他类型则可使用ArrayMap。HashMap的查找和插入时间复杂度为O(1)的代价是牺牲大量的内存来实现的,而SparseArray和ArrayMap性能略逊于HashMap,但更节省内存。

ArrayMap

ArrayMap相比HashMap内存更少,速度更慢。 ArraySet, ArrayList类似。

public final class ArrayMap implements Map {

private static final boolean CONCURRENT_MODIFICATION_EXCEPTIONS = true;

private static final int BASE_SIZE = 4; // 容量增量的最小值
private static final int CACHE_SIZE = 10; // 缓存数组的上限

static Object[] mBaseCache; //用于缓存大小为4的ArrayMap
static int mBaseCacheSize;
static Object[] mTwiceBaseCache; //用于缓存大小为8的ArrayMap
static int mTwiceBaseCacheSize;

final boolean mIdentityHashCode;
int[] mHashes; //由key的hashcode所组成的数组
Object[] mArray; //由key-value对所组成的数组,是mHashes大小的2倍
int mSize; //成员变量的个数
}

public class SparseArray implements Cloneable {
private static final Object DELETED = new Object();
private boolean mGarbage = false; //标记是否存在待回收的键值对

private int[] mKeys;
private Object[] mValues;
private int mSize;
}

SparseArray使用二分查找来找到key对应的插入位置,保证mKeys数组从小到大的排序。

SparseArray对应的key只能是int类型,它不会对key进行装箱操作。它使用了两个数组,一个保存key,一个保存value。 从内存使用上来说,SparseArray不需要保存key所对应的哈希值,所以比ArrayMap还能再节省1/3的内存。

image

SparseArray,key是int类型的Map,更加memory-efficient。

SparseArray

如果当前数组内容已填充满时,则会先进行扩容,再通过System.arraycopy来进行数据拷贝,最后在相应位置写入数据。

将修改已有键值对和插入新的键值对合在一个put()方法中,主要是依赖indexOf()过程中采用的二分查找法,在mHashes数组中查找值等于hash的key,当找到相应key时则返回正值,但找不到key则返回负值,按位取反所对应的值代表的是需要插入的位置index。

Put原理

为了减少频繁地创建和回收,特意设计了两个缓存池,分别缓存大小为4和8的ArrayMap对象。

缓存

  • mHashes是一个记录所有key的hashcode值组成的数组,是从小到大的排序方式;

  • mArray是一个记录着key-value键值对所组成的数组,是mHashes大小的2倍;

image

用一句话总结就是ArrayMap使用两个数组进行数据存储,一个int[]数组,用于保存每个item的hashCode. 一个Object[]数组,保存key/value键值对。容量是上一个数组的两倍

你可能感兴趣的:(Android知识点 ArrayMap SparseArray)