Java学习者论坛

 找回密码
 立即注册

QQ登录

只需一步,快速开始

手机号码,快捷登录

恭喜Java学习者论坛(https://www.javaxxz.com)已经为数万Java学习者服务超过8年了!积累会员资料超过10000G+
成为本站VIP会员,下载本站10000G+会员资源,购买链接:点击进入购买VIP会员
JAVA高级面试进阶视频教程Java架构师系统进阶VIP课程

分布式高可用全栈开发微服务教程

Go语言视频零基础入门到精通

Java架构师3期(课件+源码)

Java开发全终端实战租房项目视频教程

SpringBoot2.X入门到高级使用教程

大数据培训第六期全套视频教程

深度学习(CNN RNN GAN)算法原理

Java亿级流量电商系统视频教程

互联网架构师视频教程

年薪50万Spark2.0从入门到精通

年薪50万!人工智能学习路线教程

年薪50万!大数据从入门到精通学习路线年薪50万!机器学习入门到精通视频教程
仿小米商城类app和小程序视频教程深度学习数据分析基础到实战最新黑马javaEE2.1就业课程从 0到JVM实战高手教程 MySQL入门到精通教程
查看: 614|回复: 0

[android] Android内存优化之内存缓存

[复制链接]

该用户从未签到

发表于 2015-8-20 11:04:00 | 显示全部楼层 |阅读模式
什么是缓存
缓存技术原理就是把用户访问的所有对象看作一个全集,经过算法标记哪些是用户经常访问的对象,把这些对象放到一个集合里,这个集合是全集一个子集,下一次用户再访问的时候会先从这个子集集合中查找用户要访问的对象如果找到就直接返回这个对象,如果没有找到则再去全集中查找。当然了我这里说的只是原理性的东西,缓存是有很多算法的,并且有的不止一级缓存,这里就不过多讲了。
为什么要用到缓存?
有缓存的话可以不必每次从源地址读取文件,既节省了时间也节省了流量。尤其是手机设备,频繁的访问网络资源会消耗很多用户的流量和电量,这是用户不能忍受的,所以无论从哪个方面考虑应用程序都必须加上缓存。
Android中的图片缓存有哪些?各有什么特点?
Android设备的图片缓存分两种,一种是内存缓存,图片缓存在设备的内存中,一种是外部缓存,图片缓存在磁盘上,磁盘可以是内部的存储空间也可以是外部的sd卡。这两种缓存各有各的优点,内存缓存优点是快,缺点是因为也是读取到内存中所以也会消耗内存,所以不能太大,用的时候要考虑分配的空间,还有一个缺点是应用重启后就会消失。外部缓存的优点是可以长久保存大量的数据(相比较内存缓存而言),缺点就是慢。
内存缓存:
Android中官网推荐使用LruCache作为内存缓存,LruCache实际上就是一个LinkedHashMap( 补充知识:LinkedHashMap是一个双向循环列表,不支持线程安全,LruCache对它进行了封装添加了线程安全操作),里面保存了一定数量的对象强引用,每次添加的新对象都是在链表的头,当分配的空间用完的时候会把末尾的对象移除,移除的对象就可以被gc回收了。这里需要注意一下LruCache的容量,这个容量既不能太大,会造成OOM,又不能太小,起不到缓存的作用。Google官网给出一下意见作为参考:
·分配LruCache大小的时候考虑你的应用剩余内存有多大;
·一次屏幕显示多少张图片,有多少张图片是缓存起来准备显示的;
·考虑你的手机分辨率和尺寸,缓存相同的图片个数,dpi越大的手机需要的内存就会越大;
·图片分辨率和像素质量也决定了占用内存的大小;
·图片访问的频繁程度是多少,是不是有一些图片是经常访问的?如果存在你可以考虑用多个·LruCache来做缓存,按照访问的频率度分配到不同的LruCache中;
·如何平衡一下图片质量和数量,有些时候可以考虑缓存低分辨率的图片,用到的时候再在后台请求更高质量的图片;
·总之你分配的LruCache大小既不能太大,又不能太小,具体到应用中还要你综合考虑。
下面的代码是使用LruCache的例子:
1  privateLruCache<String, Bitmap> mMemoryCache;//声明缓存空间
2  finalint maxMemory = (int) (Runtime.getRuntime().maxMemory() / 1024);//获取应用在系统中的最大内存分配
3  //分配1/8的应用内存作为缓存空间
4  finalint cacheSize = maxMemory / 8;
5  mMemoryCache= new LruCache<String, Bitmap>(cacheSize) {
6  @Override
7  protectedint sizeOf(String key, Bitmap bitmap) {
8  //重写sizeOf方法,返回图片的占用字节数而不是图片的个数,每次添加图片是会被调用
9  returnbitmap.getByteCount() / 1024;
10        }
11  };
     
注意:有同学可能会问下面的代码:
1  intcacheSize=4*1024*1024;//4MiB
2  LruCachebitmapCache=newLruCache(cacheSize){
3  protectedintsizeOf(Stringkey,Bitmapvalue){
4  returnvalue.getByteCount();
5     }
6  }
这两个sizeOf的计算是不一样的,这里说明一下,这个方法重写的目的是返回图片占用的缓存空间而不是图片的数目,并且这个数值的单位要和cacheSize一样。
总结:
综合上面的讲解,在使用内存缓存LruCache时你需要知道如下知识:
LruCache封装了LinkedHashMap,提供了LRU(LeastRecently Used 最近最少使用算法)缓存的功能;
LruCache通过trimToSize方法自动删除最近最少访问的键值对;
LruCache不允许空键值, LinkedHashMap允许;
LruCache线程安全, LinkedHashMap线程不安全;
继承LruCache时,必须要复写sizeOf方法,用于计算每个条目的大小。在putget的时候会调用safeSizeOf(K key, V value)safeSizeOf(K key, Vvalue)会调用 sizeOf (K key, V value),这个方法默认返回1
另外推荐一款第三方的内测工具对APP进行全方面的检测:http://www.ineice.com/

回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

QQ|手机版|Java学习者论坛 ( 声明:本站资料整理自互联网,用于Java学习者交流学习使用,对资料版权不负任何法律责任,若有侵权请及时联系客服屏蔽删除 )

GMT+8, 2024-4-17 07:10 , Processed in 0.374633 second(s), 47 queries .

Powered by Discuz! X3.4

© 2001-2017 Comsenz Inc.

快速回复 返回顶部 返回列表