【Java源码分析】Android-SparseArray源码分析
类的定义
public class SparseArray<E> implements Cloneable {}
将int映射为对象,比HashMap更节省内存。一方面是避免了对key的自动装箱;另外一个方面是它的键值对不依赖于外部的实体对象来保存键值映射而HashMap需要额外的存储空间来保存键值对之间的映射关系。
- 该容器将映射存储在一个数组数据结构中,使用二分查找找到对应的key.这种设计不太适合于存储大量的数据。由于查找的时候需要进行二分查找,添加和删除的时候需要插入和删除数组中的实体,因此时间效率上并不高效。在只有几百个键值对的情况下,性能损失低于Hashmap的50%
- 为了弥补性能上的损失,在删除元素的时候容器实现了一个优化。仅仅将该实体标记为已删除,而不是在数组中执行实际的删除操作。这样被标记为删除的实体,可以被相同的key对应的实体复用。或者不复用而是在一段时间之后统一进行内存的回收。这个思路有点像内存回收策略中的标记清除
- 可以使用keyAt()和valueAt()对SparseArray进行迭代
- 由于按key查找的时候使用的是二分查找,那么key是有序的,所以使用keyAt(i)的时候,对应的从大到小的i得到的key也是从大到小的
主要成员变量
private int[] mKeys;
private Object[] mValues;
private int mSize;
构造函数
public SparseArray() {this(10);
}public SparseArray(int initialCapacity) {if (initialCapacity == 0) {mKeys = EmptyArray.INT;mValues = EmptyArray.OBJECT;} else {mValues = ArrayUtils.newUnpaddedObjectArray(initialCapacity);mKeys = new int[mValues.length];}mSize = 0;
}
构造函数中如果指定initialCapacity为0,那么不会做实际的内存分配操作
get方法
public E get(int key) {return get(key, null);
}@SuppressWarnings("unchecked")
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];}
}
查找指定的key对应的值,如果无法查找到该key对应的键值对,那么返回null或者指定的值
删除
// 1
public void delete(int key) {int i = ContainerHelpers.binarySearch(mKeys, mSize, key);if (i >= 0) {if (mValues[i] != DELETED) {mValues[i] = DELETED;mGarbage = true;}}
}// 2
public void remove(int key) {delete(key);
}// 3
public E removeReturnOld(int key) {int i = ContainerHelpers.binarySearch(mKeys, mSize, key);if (i >= 0) {if (mValues[i] != DELETED) {final E old = (E) mValues[i];mValues[i] = DELETED;mGarbage = true;return old;}}return null;
}// 4
public void removeAt(int index) {if (mValues[index] != DELETED) {mValues[index] = DELETED;mGarbage = true;}
}// 5
public void removeAtRange(int index, int size) {final int end = Math.min(mSize, index + size);for (int i = index; i < end; i++) {removeAt(i);}
}
删除指定的key对应的键值对。第二个方法和第一个方法是一样的,只是名字不一样。低三个方法会返回该key对应的值。第四个是根据下标删除,前面的删除方法中都有一个二分查找对应下标的过程
添加操作
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++;}
}
添加新的键值对,如果对应的key已经存在,那么替换该key对应的旧值。注意在查找失败的时候,也就是下标小于0的时候,进行了按位取反的一个操作i = ~i;
不是很清楚这里为什么不是直接取反而是按位取反。另外在添加之后会根据实际的占用情况决定是否进行GC,注意这里的GC仅仅是进行了一次数组的紧缩操作,把标记为删除的实体都给实际的清除掉
private void gc() {// Log.e("SparseArray", "gc start with " + mSize);int n = mSize;int o = 0;int[] keys = mKeys;Object[] values = mValues;for (int i = 0; i < n; i++) {Object val = values[i];if (val != DELETED) {if (i != o) {keys[o] = keys[i];values[o] = val;values[i] = null;}o++;}}mGarbage = false;mSize = o;// Log.e("SparseArray", "gc end with " + mSize);
}
获取大小
public int size() {if (mGarbage) {gc();}return mSize;
}
由于前面描述的,SparseArray采用了一个标记删除的加速方法,所以在获取大小的时候需要先实际的清除一遍然后获取真实的大小。
根据下标获取key和value
public int keyAt(int index) {if (mGarbage) {gc();}return mKeys[index];
}@SuppressWarnings("unchecked")
public E valueAt(int index) {if (mGarbage) {gc();}return (E) mValues[index];
}
上述方法分别根据下标返回该下标对应的键值对的key或者value,注意index需要在0-size()-1范围内;此外还需要判断是否执行gc()
根据key或者value获取对应的下标
public int indexOfKey(int key) {if (mGarbage) {gc();}return ContainerHelpers.binarySearch(mKeys, mSize, key);
}public int indexOfValue(E value) {if (mGarbage) {gc();}for (int i = 0; i < mSize; i++)if (mValues[i] == value)return i;return -1;
}
两个方法实现的略微不一样,因为键值对中对键的查找要更加频繁,所以使用了二分查找,而使用值进行查找的机会比较少,所以直接使用for循环。如果查找到了对应的下标,就返回,否则返回负数
清空容器
public void clear() {int n = mSize;Object[] values = mValues;for (int i = 0; i < n; i++) {values[i] = null;}mSize = 0;mGarbage = false;
}
这里有一点不是很理解的是for循环中为什么不是mValues[i] = null;
使用一个引用values[]的意义在哪里
追加
public void append(int key, E value) {if (mSize != 0 && key <= mKeys[mSize - 1]) {put(key, value);return;}if (mGarbage && mSize >= mKeys.length) {gc();}mKeys = GrowingArrayUtils.append(mKeys, mSize, key);mValues = GrowingArrayUtils.append(mValues, mSize, value);mSize++;
}
该操作如果要插入的key介于最大的key和最小的key之间,那么直接执行普通的put操作,如果大于最大的key,直接追加到尾部
补充一直用到的二分查找
// This is Arrays.binarySearch(), but doesn't do any argument validation.
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; // value found}}return ~lo; // value not present
}
比较基本的一个二分查找算法,数据结构上都有的
【Java源码分析】Android-SparseArray源码分析相关推荐
- android view 源码分析,Android ViewPager源码详细分析
1.问题 由于Android Framework源码很庞大,所以读源码必须带着问题来读!没有问题,创造问题再来读!否则很容易迷失在无数的方法与属性之中,最后无功而返. 那么,关于ViewPager有什 ...
- android应用内存分析,Android应用程序内存分析-Memory Analysis for Android Applications
Android应用程序内存分析 原文链接:http://android-developers.blogspot.com/2011/03/memory-analysis-for-android.html ...
- adb android源码分析,Android Adb 源码解析(base on Android 9.0)
Adb 框架 Adb架构 Android Adb 一共分为三个部分:adb.adb server.adbd,源码路径:system/core/adb. adb和adb server 是运行在P ...
- android 3d城市源码,[转载]android Gallery3D源码分析
一.布局 gallery3d的界面生成和普通的应用程序不一样.普通程序一般一个界面就是一个activity,布局用xml或代码都可以实现,界面切换是activity的切换方式:而gallery3d没有 ...
- android launcher3源码分析,Android Launcher3源码分析与修改
Launcher和Setting是客户需求经常改动的地方,不过其代码量也不容小觑.今天就初略来看一下,以下内容都是本人查阅资料加上自己的理解得出,由于自己水平有限,如果误导还请指出: 先从Androi ...
- android联系人源码分析,android 联系人源码分析 新字段的添加流程
android 联系人 代码主要分布在四个地方(有的代码会有关于sim卡的加载,各个平台实现方式不同,就不提了): 1 framwork/base/core/java/android/provider ...
- java做app流程图,Android App 构建流程分析
我们平时在android studio中点击run ,就能把代码编译成一个apk文件并安装到手机上.那么这个过程中都具体发生了什么 ?我们是怎么把代码和资源文件打包成一个apk文件,并安装到手机上的呢 ...
- 二维码扫描器 android二维码扫描 二维码识别软件
这两天,在弄一个二维码扫描的东西. 二维码为移动设备购物提供了方便.在电商行业也有一些商家开始正式商业应用,我所知道的 1号店就在使用这种东西. 一个二维码对应一个产品. 二维码可以使用Google提 ...
- android 图片分析,Android图片处理实例分析
本文实例讲述了Android图片处理的方法.分享给大家供大家参考,具体如下: package cn.szbw.util; import Android.content.Context; import ...
- android内存占用分析,Android App性能评测分析-内存篇
1.内存了解 在Android App的性能优化的各个部分里,内存方面的知识较多且不易理解,内存的问题绝对是最令人头疼的一部分,需要对内存基础知识.内存分配.内存管理机制等非常熟悉,才能排查问题. 1 ...
最新文章
- python安装教程32位-python为什么要装32位的
- Spring源码解析-核心类之XmlBeanDefinitionReader
- Redis的Expire与Setex
- mysql企业版多少钱_企业资质代办多少钱
- boost::pointer_traits的用法实例
- openflow多级流表机制的优点?
- php发送验证码短信,php发送短信验证码
- openvino量化自己训练的yolov3模型至int8(有成功验证截图)
- MYSQL-skip-networking
- iOS腾讯百度面试题
- 推理集 —— 特殊的时间
- 京瓷打印机p5026cdn_京瓷ECOSYS P5026cdn驱动
- 超子的水族箱升级计划
- GAMIT处理GLONASS数据
- 服务器监控报警系统软件设计,Monitor监控报警系统
- Java身份证、手机号码用*隐藏中间几位
- 《今日简史》--重新认识自己:人类心智的奥秘
- Flask之解读app.py文件
- Graham-Scan算法计算凸包的Python代码实现
- webrtc入门之客户端连麦demo-apprtc
热门文章
- php禁止伪造_php防止伪造的数据从URL提交方法
- matlab神经网络 时间序列,请问吧里有大神做过MATLAB时间序列神经网络(NARX)吗?...
- 支持多种小程序!阿里云ARMS推出小程序监控
- 「北京」「10-30k」「华米科技(小米手环)」招前端工程师
- 监控io性能, free命令, ps命令, 查看网络状态, linux下抓包
- layui upload 额外参数上传
- 浪潮成立人工智能部门 为AI提供顶尖计算产品
- 最先进数据中心都建在哪?
- Linux(Centos)快速搭建SVN服务器
- FreePBX SIP Trunk