使用 ArrayMap And SparseArray 来优化你的 Android 应用

   2017-01-06 0
核心提示:这篇文章将说明为什么使用 ArrayMap 和 SparseArray 优化你的 Android 应用。每当你需要存储 key -value 键值对数据时,第一个的想到的数据结构应该就是 HashMap .它是非常灵活的,我们到处都在使用它,而没有考虑它所带来的副作用.如果你使用 Android Studio 开

这篇文章将说明为什么使用 ArrayMap 和 SparseArray 优化你的 Android 应用。

每当你需要存储 key ->value 键值对数据时,第一个的想到的数据结构应该就是 HashMap .它是非常灵活的,我们到处都在使用它,而没有考虑它所带来的副作用.

如果你使用 Android Studio 开发你的应用,当你在项目中使用 HashMap 时它将会警告你使用 ArrayMap 来替代它.所以让我们来探讨为什么你应该使用 ArrayMap,和它的内部是如何工作的.

HashMap VS ArrayMap

HashMap来自 java.util.HashMap 包 ,ArrayMap 来自 amdrpod util.ArrayMap 和 android.support.v4.util.ArrayMap 包,提供support.v4来支持较低安卓的版本.

这个 官方视频 可以让你更进一步的了解它的细节.

ArrayMap是一个通用的 key->value 映射数据结构的,它的设计可以更有效的节省内存,ArrayMap保持它的映射到一个数组数据结构,一个 Integer 数组来存储每一个 item 的哈希码,一个对象数组来存储键值对,这样做避免了为每一个条目都创建一个额外的对象,它也更有效的控制了数组的大小.它自增长的复制你的存储的数据到数组-没有重建一个映射.

请注意,这个数据结构实现的目的不是为了要包含大量的条目.它通常比传统的 HashMap 慢,因为需要使用二分查找.添加和删除操作需要插入或者移除数组中的实例.

HashMap

HashMap 是一个基于 HashMap.Entry 数组对象( Entry 是它的一个内部类)。

Entry 类的实例对象包含:

  • 一个泛型key
  • 一个泛型 value
  • 对象的哈希码
  • 指向下一个Entry 的指针

当我们将一个键值对插入到 HashMap 会发生什么?

  • 根据 key 计算哈希码
  • 如果没碰撞直接放到bucket里
  • 如果碰撞了,以链表的形式存在buckets后
  • 如果碰撞导致链表过长(大于等于TREEIFY_THRESHOLD),就把链表转换成红黑树
  • 如果插入的数据是预先存在的,则将当前元素替换之前的元素.(保证 key 唯一).

现在当你根据key 查询hashMap 中的 value ,它将花费 O(1) 时间,但是重要的是它花费了更多的内存空间.

缺点:

  • 插入每一个实例需要创建一个额外的对象,这将影响内存使用,以及垃圾收集. HashMap.Entry 本身是一个额外的对象需要被垃圾回收.
  • HashMap 每一次操作都需要压缩或者扩充.这随着对象的增长这是一个很昂贵的操作.
  • 在 Android 内存是一个很重要的,连续分配和回收内存,随着垃圾回收期,将导致你的 Android 应用延迟.

记住,垃圾回收对于 Android 应用的性能是一个重负.当垃圾回收期正在进行,你的应用将不能运行.

ArrayMap

ArrayMap 使用两个数组.

这个对象使用内部的一个 Object[ ] mArray 存储数据, int[] mHashes 来存储哈希码.

当插入一个键值对:

  • 这个键值对将自动装箱.
  • key 插入到 mArray 可用的下一个位置.
  • value 插入到 key 所在的下一个位置.
  • 根据 key 计算哈希码,将他放置在 mHashes[ ]的下一个可用的位置.

当你查找这个 key :

  • 计算这个key 的哈希码,使用二分查找哈希码在 mHashes所在的索引.
  • 一旦我们根据哈希码得到索引,我们就知道了key 在 mArray的2 index 位置, value 在 mArray 的2 index+1的位置.此时,时间复杂度从O(1) 到 O(logN),但是它非常节省内存.
  • 当你存储100条数据,你感受不到任何的延迟.但是这将使你的应用程序更加的高效.

推荐数据结构:

ArrayMap<K,V> 替换 HashMap<K,V>
ArraySet<K,V> 替换 HashSet<K,V>
SparseArray 替换 HashMap<Integer,V>
SparseBooleanArray 替换 HashMap<Integer,Boolean>
SparseIntArray  替换 HashMap<Integer,Integer>
SparseLongArray 替换 HashMap<Long,V>
LongSparseArray<V> 替换 HashMap<Long,V>

原文链接: Android App Optimization Using ArrayMap and SparseArray

 
标签: 安卓开发
反对 0举报 0 评论 0
 

免责声明:本文仅代表作者个人观点,与乐学笔记(本网)无关。其原创性以及文中陈述文字和内容未经本站证实,对本文以及其中全部或者部分内容、文字的真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
    本网站有部分内容均转载自其它媒体,转载目的在于传递更多信息,并不代表本网赞同其观点和对其真实性负责,若因作品内容、知识产权、版权和其他问题,请及时提供相关证明等材料并与我们留言联系,本网站将在规定时间内给予删除等相关处理.

  • 安卓中通知功能的具体实现
    安卓中通知功能的具体实现
    通知[Notification]是Android中比较有特色的功能,当某个应用程序希望给用户发出一些提示信息,而该应用程序又不在前台运行时,就可以借助通知实现。使用通知的步骤1、需要一个NotificationManager来获得NotificationManager manager = (NotificationManager
    02-05 安卓开发
  • Android view系统分析-setContentView
    Android view系统分析-setContentView
    第一天上班,列了一下今年要学习的东西。主要就是深入学习Android相关的系统源代码,夯实基础。对于学习Android系统源代码,也没什么大概,就从我们平常使用最基础的东西学起,也就是从view这个切入点开始学习Android的源码,在没分析源码之前,我们有的时候
    02-05 安卓开发
  • 如何进行网络视频截图/获取视频的缩略图
    如何进行网络视频截图/获取视频的缩略图
    小编导读:获取视频的缩略图,截图正在播放的视频某一帧,是在音视频开发中,常遇到的问题。本文是主要用于点播中截图视频,同时还可以获取点播视频的缩略图进行显示,留下一个问题,如下图所示, 如果要获取直播中节目视频缩略图,该怎么做呢?(ps:直播是直
  • Android NDK 层发起 HTTP 请求的问题及解决
    Android NDK 层发起 HTTP 请求的问题及解决
    前言新的一年,大家新年快乐~~鸡年大吉!本次给大家带来何老师的最新文章~虽然何老师还在过节,但依然放心不下广大开发者,在此佳节还未结束之际,给大家带来最新的技术分享~ 事件的起因不说了,总之是需要实现一个 NDK 层的网络请求。为了多端适用,还是选择
  • Android插件化(六): OpenAtlasの改写aapt以防止资源ID冲突
    Android插件化(六): OpenAtlasの改写aapt以防
    引言Android应用程序的编译中,负责资源打包的是aapt,如果不对打包后的资源ID进行控制,就会导致插件中的资源ID冲突。所以,我们需要改写aapt的源码,以达到通过某种方式传递资源ID的Package ID,通过aapt打包时获取到这个Package ID并且应用才插件资源的命名
    02-05 安卓开发
  • Android架构(一)MVP架构在Android中的实践
    Android架构(一)MVP架构在Android中的实践
    为什么要重视程序的架构设计 对程序进行架构设计的原因,归根结底是为了 提高生产力 。通过设计是程序模块化,做到模块内部的 高聚合 和模块之间的 低耦合 (如依赖注入就是低耦合的集中体现)。 这样做的好处是使得程序开发过程中,开发人员主需要专注于一点,
    02-05 安卓开发
  • 安卓逆向系列教程 4.2 分析锁机软件
    安卓逆向系列教程 4.2 分析锁机软件
    安卓逆向系列教程 4.2 分析锁机软件 作者: 飞龙 这个教程中我们要分析一个锁机软件。像这种软件都比较简单,完全可以顺着入口看下去,但我这里还是用关键点来定位。首先这个软件的截图是这样,进入这个界面之后,除非退出模拟器,否则没办法回到桌面。上面那
    02-05 安卓开发
  • Android插件化(二):OpenAtlas插件安装过程分析
    Android插件化(二):OpenAtlas插件安装过程分析
    在前一篇博客 Android插件化(一):OpenAtlas架构以及实现原理概要 中,我们对应Android插件化存在的问题,实现原理,以及目前的实现方案进行了简单的叙述。从这篇开始,我们要深入到OpenAtlas的源码中进行插件安装过程的分析。 插件的安装分为3种:宿主启动时立
    02-05 安卓开发
  • [译] Android API 指南
    [译] Android API 指南
    众所周知,Android开发者有中文网站了,API 指南一眼看去最左侧的菜单都是中文,然而点进去内容还是很多是英文,并没有全部翻译,我这里整理了API 指南的目录,便于查看,如果之前还没有通读,现在可以好好看一遍。注意,如果标题带有英文,说明官方还没有翻
  • 使用FileProvider解决file:// URI引起的FileUriExposedException
    使用FileProvider解决file:// URI引起的FileUri
    问题以下是一段简单的代码,它调用系统的相机app来拍摄照片:void takePhoto(String cameraPhotoPath) {File cameraPhoto = new File(cameraPhotoPath);Intent takePhotoIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);takePhotoIntent.putExtra(Medi
    02-05 安卓开发
点击排行