关于ANDROID示例程序(BITMAPFUN)——高效加载图片的坑爹地方

本文涉及的产品
日志服务 SLS,月写入数据量 50GB 1个月
简介:

下面的都是费话,不想看的,直接看红色字体,然后自己实验下

  在android的开发指南上有这样一篇文章,如何更有效率的加载图片,地址为 https://developer.android.com/training/displaying-bitmaps/index.html,这篇文章详细地介绍了如何加载高清图到内存,同时避免系统报OOM的问题,文章写得很不错,示例程序也可以直接运行。在我们项目的一次小版本升级的过程中,我们尝试了使用git上的一个开源项目afinal(bitmapfun的封装版)来加载图片,但是在测试的时候发现了一个问题,新的图片加载器(bitmapfun)比之前用的ImageDownloader要慢很多,特别是在网络状况不好的时候,那简直是坑爹,等5s钟算少的,一般要等10s左右,老大找到我,说这个图片加载不出来啊,太慢了。我靠,不是吧,这玩笑开的有点大了吧,拿来一用,果然是慢很多。

  然后开始了调试工作,开始我在想是不是网络下载部分不一样,之前用的是httpclient,现在bitmapfun用的是url.openConnection,果断替换成之前的httpclient方法,感觉有所好转(其实是自己心理在作怪),再试试,发现有时快,有时慢,对比之前的,之前的图片一直很稳定,不会出现这种情况啊,然后想了个办法,在项目中的找了两个页面,一个用之前的ImageDownload加载,另一个用bitmapfun加载,然后把各个时段(可以分为3段吧,1是从内存缓存中查找、2是从网络加载、3是存到内在缓存和sdcard缓存)的时间打印出来对比,发现确实是bitmapfun下载这块(processBitmap方法)里面最耗时间,这部分的代码,看到同步锁,应该是他吧,咋一看,没问题啊,只锁住一小块,应该是防止缓存没被初始化的吧,过。。。再往下看,频繁的对sdcard进行io操作,恩对,应该是这里,把刚才的方法重复一下,分段打印耗时,一看结果,tnnd,占用时间最长的居然是wait操作,也就是那个锁的等待,再去看下代码,双击下锁的范围,发现基本是对整个下载过程进行了同步锁定,这尼玛,太坑爹了吧,一个图片在下载的时候,其他的都得在那等着,那前面新建的3个核心线程,被你同步成一个,我说怎么看到这个加载图片这么整齐,一个个出现。

  发现了问题,问下为什么,他这么做的原因是什么,其实就是为了那个DiskLruCache的日志文件,不允许多个线程同时操作,否则日志会错乱,就没办法统计哪个文件是最久未被使用的。自己想试着改下锁的范围,发现有点困难,因为下载过程中对日志文件操作的太频繁了,哪位大牛有好的方法告诉下我,不胜感激。

  对使用bitmapfun或者afinal的一点建议,DiskLruCache慢的关键在于对日志文件(目录下journal文件)的要求太高了,日志文件的作用就是记录每一个文件的访问次数,所以它每一次读取和写入都要写入日志记文件,这样是可以更准确的统计出最久未被使用的文件,但是代价太高了(频繁的io操作和同步锁),记得上个版本的bitmapfun是不用日志文件的,直接从程序访问sdcard缓存开始计算访问时间,其实这样更合适一些,对性能更好。所以如果使用这个示例程序来加载图片,最好的方式就是使用上个版本的DiskLruCache或者自己想办法来实现日志文件的记录,有更好的方法包括日志文件的记录方法,记得告诉我哦!

相关实践学习
日志服务之使用Nginx模式采集日志
本文介绍如何通过日志服务控制台创建Nginx模式的Logtail配置快速采集Nginx日志并进行多维度分析。
相关文章
|
4月前
|
Java Android开发
Android面试题经典之Glide取消加载以及线程池优化
Glide通过生命周期管理在`onStop`时暂停请求,`onDestroy`时取消请求,减少资源浪费。在`EngineJob`和`DecodeJob`中使用`cancel`方法标记任务并中断数据获取。当网络请求被取消时,`HttpUrlFetcher`的`cancel`方法设置标志,之后的数据获取会返回`null`,中断加载流程。Glide还使用定制的线程池,如AnimationExecutor、diskCacheExecutor、sourceExecutor和newUnlimitedSourceExecutor,其中某些禁止网络访问,并根据CPU核心数动态调整线程数。
138 2
|
26天前
|
Android开发 UED
Android 中加载 Gif 动画
【10月更文挑战第20天】加载 Gif 动画是 Android 开发中的一项重要技能。通过使用第三方库或自定义实现,可以方便地在应用中展示生动的 Gif 动画。在实际应用中,需要根据具体情况进行合理选择和优化,以确保用户体验和性能的平衡。可以通过不断的实践和探索,进一步掌握在 Android 中加载 Gif 动画的技巧和方法,为开发高质量的 Android 应用提供支持。
|
2月前
|
存储 缓存 编解码
Android经典面试题之图片Bitmap怎么做优化
本文介绍了图片相关的内存优化方法,包括分辨率适配、图片压缩与缓存。文中详细讲解了如何根据不同分辨率放置图片资源,避免图片拉伸变形;并通过示例代码展示了使用`BitmapFactory.Options`进行图片压缩的具体步骤。此外,还介绍了Glide等第三方库如何利用LRU算法实现高效图片缓存。
65 20
Android经典面试题之图片Bitmap怎么做优化
|
6月前
|
Android开发
Android通过手势(多点)缩放和拖拽图片
Android通过手势(多点)缩放和拖拽图片
53 4
|
6月前
|
Java Android开发
android 下载图片的问题
android 下载图片的问题
45 3
|
3月前
|
数据处理 开发工具 数据安全/隐私保护
Android平台RTMP推送|轻量级RTSP服务|GB28181接入之文字、png图片水印的精进之路
本文探讨了Android平台上推流模块中添加文字与PNG水印的技术演进。自2015年起,为了满足应急指挥及安防领域的需求,逐步发展出三代水印技术:第一代为静态文字与图像水印;第二代实现了动态更新水印内容的能力,例如实时位置与时间信息;至第三代,则优化了数据传输效率,直接使用Bitmap对象传递水印数据至JNI层,减少了内存拷贝次数。这些迭代不仅提升了用户体验和技术效率,也体现了开发者追求极致与不断创新的精神。
|
3月前
|
自然语言处理 定位技术 API
Android经典实战之如何获取图片的经纬度以及如何根据经纬度获取对应的地点名称
本文介绍如何在Android中从图片提取地理位置信息并转换为地址。首先利用`ExifInterface`获取图片内的经纬度,然后通过`Geocoder`将经纬度转为地址。注意操作需在子线程进行且考虑多语言支持。
232 4
|
3月前
|
存储 缓存 Java
Android项目架构设计问题之优化业务接口数据的加载效率如何解决
Android项目架构设计问题之优化业务接口数据的加载效率如何解决
45 0
|
3月前
|
Java Android开发 Kotlin
Android项目架构设计问题之要在Glide库中加载网络图片到ImageView如何解决
Android项目架构设计问题之要在Glide库中加载网络图片到ImageView如何解决
37 0
|
3月前
|
XML 前端开发 Android开发
Android经典实战之Kotlin中实现圆角图片和圆形图片
本文介绍两种实现圆角图像视图的方法。第一种是通过自定义Kotlin `AppCompatImageView`,重写`onDraw`方法使用`Canvas`和`Path`进行圆角剪裁。第二种利用Android Material库中的`ShapeableImageView`,简单配置即可实现圆角效果。两种方法均易于实现且提供动态调整圆角半径的功能。
73 0
下一篇
无影云桌面