Android异步加载图片并缓存到内存和SD卡上

2024-03-10 17:18

本文主要是介绍Android异步加载图片并缓存到内存和SD卡上,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

* 异步加载图片* 使用方法:* private AsyncImageLoader asyImg = new AsyncImageLoader();* asyImg.LoadImage(productItems.get(position).getPic(), (ImageView)view.findViewById(R.id.pic));*/public class AsyncImageLoader {// 为了加快速度,在内存中开启缓存(主要应用于重复图片较多时,或者同一个图片要多次被访问,比如在ListView时来回滚动)public Map<String, SoftReference<Drawable>> imageCache = new HashMap<String, SoftReference<Drawable>>();private ExecutorService executorService = Executors.newFixedThreadPool(5); // 固定五个线程来执行任务private final Handler handler = new Handler();// SD卡上图片储存地址private final String path = Environment.getExternalStorageDirectory().getPath() + "/maiduo";/*** * @param imageUrl*            图像url地址* @param callback*            回调接口* @return 返回内存中缓存的图像,第一次加载返回null*/public Drawable loadDrawable(final String imageUrl,final ImageCallback callback) {// 如果缓存过就从缓存中取出数据if (imageCache.containsKey(imageUrl)) {SoftReference<Drawable> softReference = imageCache.get(imageUrl);if (softReference.get() != null) {return softReference.get();}} else if (useTheImage(imageUrl) != null) {return useTheImage(imageUrl);}// 缓存中没有图像,则从网络上取出数据,并将取出的数据缓存到内存中executorService.submit(new Runnable() {public void run() {try {final Drawable drawable = Drawable.createFromStream(new URL(imageUrl).openStream(), "image.png");imageCache.put(imageUrl, new SoftReference<Drawable>(drawable));handler.post(new Runnable() {public void run() {callback.imageLoaded(drawable);}});saveFile(drawable, imageUrl);} catch (Exception e) {throw new RuntimeException(e);}}});return null;}// 从网络上取数据方法public Drawable loadImageFromUrl(String imageUrl) {try {return Drawable.createFromStream(new URL(imageUrl).openStream(),"image.png");} catch (Exception e) {throw new RuntimeException(e);}}// 对外界开放的回调接口public interface ImageCallback {// 注意 此方法是用来设置目标对象的图像资源public void imageLoaded(Drawable imageDrawable);}// 引入线程池,并引入内存缓存功能,并对外部调用封装了接口,简化调用过程public void LoadImage(final String url, final ImageView iv) {if (iv.getImageMatrix() == null) {iv.setImageResource(R.drawable.loading);}// 如果缓存过就会从缓存中取出图像,ImageCallback接口中方法也不会被执行Drawable cacheImage = loadDrawable(url,new AsyncImageLoader.ImageCallback() {// 请参见实现:如果第一次加载url时下面方法会执行public void imageLoaded(Drawable imageDrawable) {iv.setImageDrawable(imageDrawable);}});if (cacheImage != null) {iv.setImageDrawable(cacheImage);}}/*** 保存图片到SD卡上* * @param bm* @param fileName* */public void saveFile(Drawable dw, String url) {try {BitmapDrawable bd = (BitmapDrawable) dw;Bitmap bm = bd.getBitmap();// 获得文件名字final String fileNa = url.substring(url.lastIndexOf("/") + 1,url.length()).toLowerCase();File file = new File(path + "/image/" + fileNa);// 创建图片缓存文件夹boolean sdCardExist = Environment.getExternalStorageState().equals(android.os.Environment.MEDIA_MOUNTED); // 判断sd卡是否存在if (sdCardExist) {File maiduo = new File(path);File ad = new File(path + "/image");// 如果文件夹不存在if (!maiduo.exists()) {// 按照指定的路径创建文件夹maiduo.mkdir();// 如果文件夹不存在} else if (!ad.exists()) {// 按照指定的路径创建文件夹ad.mkdir();}// 检查图片是否存在if (!file.exists()) {file.createNewFile();}}BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream(file));bm.compress(Bitmap.CompressFormat.JPEG, 100, bos);bos.flush();bos.close();} catch (Exception e) {// TODO: handle exception}}/*** 使用SD卡上的图片* */public Drawable useTheImage(String imageUrl) {Bitmap bmpDefaultPic = null;// 获得文件路径String imageSDCardPath = path+ "/image/"+ imageUrl.substring(imageUrl.lastIndexOf("/") + 1,imageUrl.length()).toLowerCase();File file = new File(imageSDCardPath);// 检查图片是否存在if (!file.exists()) {return null;}bmpDefaultPic = BitmapFactory.decodeFile(imageSDCardPath, null);if (bmpDefaultPic != null || bmpDefaultPic.toString().length() > 3) {Drawable drawable = new BitmapDrawable(bmpDefaultPic);return drawable;} elsereturn null;}}


这篇关于Android异步加载图片并缓存到内存和SD卡上的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/794938

相关文章

浅析Spring如何控制Bean的加载顺序

《浅析Spring如何控制Bean的加载顺序》在大多数情况下,我们不需要手动控制Bean的加载顺序,因为Spring的IoC容器足够智能,但在某些特殊场景下,这种隐式的依赖关系可能不存在,下面我们就来... 目录核心原则:依赖驱动加载手动控制 Bean 加载顺序的方法方法 1:使用@DependsOn(最直

Android kotlin中 Channel 和 Flow 的区别和选择使用场景分析

《Androidkotlin中Channel和Flow的区别和选择使用场景分析》Kotlin协程中,Flow是冷数据流,按需触发,适合响应式数据处理;Channel是热数据流,持续发送,支持... 目录一、基本概念界定FlowChannel二、核心特性对比数据生产触发条件生产与消费的关系背压处理机制生命周期

Android ClassLoader加载机制详解

《AndroidClassLoader加载机制详解》Android的ClassLoader负责加载.dex文件,基于双亲委派模型,支持热修复和插件化,需注意类冲突、内存泄漏和兼容性问题,本文给大家介... 目录一、ClassLoader概述1.1 类加载的基本概念1.2 android与Java Class

基于Python实现一个图片拆分工具

《基于Python实现一个图片拆分工具》这篇文章主要为大家详细介绍了如何基于Python实现一个图片拆分工具,可以根据需要的行数和列数进行拆分,感兴趣的小伙伴可以跟随小编一起学习一下... 简单介绍先自己选择输入的图片,默认是输出到项目文件夹中,可以自己选择其他的文件夹,选择需要拆分的行数和列数,可以通过

怎样通过分析GC日志来定位Java进程的内存问题

《怎样通过分析GC日志来定位Java进程的内存问题》:本文主要介绍怎样通过分析GC日志来定位Java进程的内存问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、GC 日志基础配置1. 启用详细 GC 日志2. 不同收集器的日志格式二、关键指标与分析维度1.

Java内存分配与JVM参数详解(推荐)

《Java内存分配与JVM参数详解(推荐)》本文详解JVM内存结构与参数调整,涵盖堆分代、元空间、GC选择及优化策略,帮助开发者提升性能、避免内存泄漏,本文给大家介绍Java内存分配与JVM参数详解,... 目录引言JVM内存结构JVM参数概述堆内存分配年轻代与老年代调整堆内存大小调整年轻代与老年代比例元空

利用Python脚本实现批量将图片转换为WebP格式

《利用Python脚本实现批量将图片转换为WebP格式》Python语言的简洁语法和库支持使其成为图像处理的理想选择,本文将介绍如何利用Python实现批量将图片转换为WebP格式的脚本,WebP作为... 目录简介1. python在图像处理中的应用2. WebP格式的原理和优势2.1 WebP格式与传统

Spring如何使用注解@DependsOn控制Bean加载顺序

《Spring如何使用注解@DependsOn控制Bean加载顺序》:本文主要介绍Spring如何使用注解@DependsOn控制Bean加载顺序,具有很好的参考价值,希望对大家有所帮助,如有错误... 目录1.javascript 前言2. 代码实现总结1. 前言默认情况下,Spring加载Bean的顺

Android DataBinding 与 MVVM使用详解

《AndroidDataBinding与MVVM使用详解》本文介绍AndroidDataBinding库,其通过绑定UI组件与数据源实现自动更新,支持双向绑定和逻辑运算,减少模板代码,结合MV... 目录一、DataBinding 核心概念二、配置与基础使用1. 启用 DataBinding 2. 基础布局

Android ViewBinding使用流程

《AndroidViewBinding使用流程》AndroidViewBinding是Jetpack组件,替代findViewById,提供类型安全、空安全和编译时检查,代码简洁且性能优化,相比Da... 目录一、核心概念二、ViewBinding优点三、使用流程1. 启用 ViewBinding (模块级