最新android-bitmap的缓存策略,2024年最新安卓面试java基础

总结

笔者之前工作是在金融公司可能并不是特别追求技术,而笔者又是喜欢追求技术的人,所以格格不入,只能把目标放在互联网大厂了。也希望大家都去敢于尝试和追逐自己的梦想!
BATJ大厂Android高频面试题

觉得有收获的记得点赞,关注+收藏哦!你们的点赞就是我的动力!

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化学习资料的朋友,可以戳这里获取

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

}

所以在lrucache中应该选择accessorder = true,当我们调用put、get方法时,linkedhashmap内部会将这个item移动到链表的尾部,即在链表尾部是最近刚刚使用的item,链表头部就是最近最少使用的item。当缓存空间不足时,可以remove头部结点释放缓存空间。
下面举例lrucache的典型使用姿势:

int maxmemory = (int) (runtime.getruntime().maxmemory() / 1024);

int cachesize = maxmemory / 8;

mmemorycache = new lrucache(cachesize) {

@override

protected int sizeof(string key, bitmap bitmap) {

return bitmap.getrowbytes() * bitmap.getheight() / 1024;

}

};


// 向 lrucache 中添加一个缓存对象

private void addbitmaptomemorycache(string key, bitmap bitmap) {

if (getbitmapfrommemcache(key) == null) {

mmemorycache.put(key, bitmap);

}

}

//获取一个缓存对象

private bitmap getbitmapfrommemcache(string key) {

return mmemorycache.get(key);

}

上述示例代码中,总容量的大小是当前进程的可用内存的八分之一(官方推荐是八分之一哈,你们可以自己视情况定),sizeof()方法计算了bitmap的大小,sizeof方法默认返回的是你缓存item数目,源码中直接return 1(这里的源码比较简单,可以自己看看~)。

如果你需要cache中某个值释放,可以重写entryremoved()方法,这个方法会在元素被put或者remove的时候调用,源码默认是空实现。重写entryremoved()方法还可以实现二级内存缓存,进一步提高性能。思路如下:重写entryremoved(),把删除掉的item,再次存入另一个linkedhashmap中。这个数据结构当做二级缓存,每次获得图片的时候,按照一级缓存 、二级缓存、sdcard、网络的顺序查找,找到就停止。

###2.disklrucache
当我们需要存大量图片的时候,我们指定的缓存空间可能很快就用完了,lrucache会频繁地进行trimtosize操作将最近最少使用的数据remove掉,但是hold不住过会又要用这个数据,又从网络download一遍,为此有了disklrucache,它可以保存这些已经下载过的图片。当然,从磁盘读取图片的时候要比内存慢得多,并且应该在非ui线程中载入磁盘图片。disklrucache顾名思义,实现存储设备缓存,即磁盘缓存,它通过将缓存对象写入文件系统从而实现缓存效果。

ps: 如果缓存的图片经常被使用,可以考虑使用contentprovider。
disklrucache的实现原理:
lrucache采用的是linkedhashmap这种数据结构来保存缓存中的对象,那么对于disklrucache呢?由于数据是缓存在本地文件中,相当于是持久保存的一个文件,即使app kill掉,这些文件还在滴。so , 到底是啥?disklrucache也是采用linekedhashmap这种数据结构,但是不够,需要加持buff

日志文件。日志文件可以看做是一块“内存”,map中的value只保存文件的简要信息,对缓存文件的所有操作都会记录在日志文件中。

disklrucache的初始化:

下面是disklrucache的创建过程:

private static final long disk_cache_size = 1024 * 1024 * 50; //50mb

file diskcachedir = getdiskcachedir(mcontext, “bitmap”);

if (!diskcachedir.exists()) {

diskcachedir.mkdirs();

}

if (getusablespace(diskcachedir) > disk_cache_size) {

try {

mdisklrucache = disklrucache.open(diskcachedir, 1, 1,

disk_cache_size);

} catch (ioexception e) {

e.printstacktrace();

}

}

瞅了一眼,可以知道重点在open()函数,其中第一个参数表示文件的存储路径,缓存路径可以是sd卡上的缓存目录,具体是指/sdcard/android/data/package_name/cache,package_name表示当前应用的包名,当应用被卸载后, 此目录会一并删除掉。如果你希望应用卸载后,这些缓存文件不被删除,可以指定sd卡上其他目录。第二个参数表示应用的版本号,一般设为1即可。第三个参数表示单个结点所对应数据的个数,一般设为1。第四个参数表示缓存的总大小,比如50mb,当缓存大小超过这个设定值后,disklrucache会清除一些缓存保证总大小不会超过设定值
disklrucache的数据缓存与获取缓存:
数据缓存操作是借助disklrucache.editor类完成的,editor表示一个缓存对象的编辑对象。

new thread(new runnable() {

@override

public void run() {

try {

string imageurl = “http://d.url.cn/myapp/qq_desk/friendprofile_def_cover_001.png”;

string key = hashkeyfordisk(imageurl); //md5对url进行加密,这个主要是为了获得统一的16位字符

disklrucache.editor editor = mdisklrucache.edit(key); //拿到editor,往journal日志中写入dirty记录

if (editor != null) {

outputstream outputstream = editor.newoutputstream(0);

if (downloadurltostream(imageurl, outputstream)) { //downloadurltostream方法为下载图片的方法,并且将输出流放到outputstream

editor.commit(); //完成后记得commit(),成功后,再往journal日志中写入clean记录

} else {

editor.abort(); //失败后,要remove缓存文件,往journal文件中写入remove记录

}

}

mdisklrucache.flush(); //将缓存操作同步到journal日志文件,不一定要在这里就调用

} catch (ioexception e) {

推荐学习资料


  • 脑图
    360°全方位性能调优

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化学习资料的朋友,可以戳这里获取

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

需要这份系统化学习资料的朋友,可以戳这里获取](https://bbs.csdn.net/topics/618156601)**

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

  • 4
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值