一.引言
Android的每一个App通常只拥有有限的系统资源,Android设备为每个App分配的内存大小是也是有上限的,并且,针对不同的设备配置所分配的内存大小也是不一样的,最小为16MB。图片会占用大量的内存,尤其是那些超清照片。所以图片加载时做容易造成安卓内存溢出的原因,而要解决这些问题还需要很多相关知识:
1、多线程下载,线程管理。
2、多级缓存架构设计和策略,内存缓存,磁盘缓存,缓存有效性处理。
3、图片压缩,特效处理,动画处理。
4、复杂网络情况下下载图片策略,例如弱网络等。
5、内存管理,lru 算法、对象引用、GC回收等优化。
Universal ImageLoader 是很早开源的图片缓存,在早期被很多应用使用。
Universal ImageLoader的GitHub地址:https://github.com/nostra13/Android-Universal-Image-Loader
最近一次更新是在3年前,已经停止更新了。不推荐新项目中使用。
二.基本使用
1. 添加依赖
compile ‘com.nostra13.universalimageloader:universal-image-loader:1.9.5‘
2.Application初始化
// 初始化参数 ImageLoaderConfiguration config = new ImageLoaderConfiguration.Builder(context) .threadPriority(Thread.NORM_PRIORITY - 2) // 线程优先级 .denyCacheImageMultipleSizesInMemory() // 当同一个Uri获取不同大小的图片,缓存到内存时,只缓存一个。默认会缓存多个不同的大小的相同图片 .discCacheFileNameGenerator(new Md5FileNameGenerator()) // 将保存的时候的URI名称用MD5 .tasksProcessingOrder(QueueProcessingType.LIFO) // 设置图片下载和显示的工作队列排序 .writeDebugLogs() // 打印debug log .build(); // 全局初始化此配置 ImageLoader.getInstance().init(config);
3.基本使用
imageLoader.displayImage(Constants.IMAGES[position],holder.iv,options); private DisplayImageOptions options = new DisplayImageOptions.Builder() .showStubImage(R.drawable.atguigu_logo) // 设置图片下载期间显示的图片 .showImageForEmptyUri(R.drawable.atguigu_logo) // 设置图片Uri为空或是错误的时候显示的图片 .showImageOnFail(R.drawable.atguigu_logo) // 设置图片加载或解码过程中发生错误显示的图片 .cacheInMemory(true) // 设置下载的图片是否缓存在内存中 .cacheOnDisk(true) // 设置下载的图片是否缓存在SD卡中 .displayer(new RoundedBitmapDisplayer(20)) // 设置成圆角图片 .build(); // 创建配置过得DisplayImageOption对象;
三.拓展了解
ImageLoaderConfiguration是针对图片缓存的全局配置,主要有线程类、缓存大小、磁盘大小、图片下载与解析、日志方面的配置。
ImageLoader是具体下载图片,缓存图片,显示图片的具体执行类,它有两个具体的方法displayImage(...)、loadImage(...),但是其实最终他们的实现都是displayImage(...)。
DisplayImageOptions用于指导每一个Imageloader根据网络图片的状态(空白、下载错误、正在下载)显示对应的图片,是否将缓存加载到磁盘上,下载完后对图片进行怎么样的处理。
1.ImageLoaderConfiguration的配置主要是全局性的配置,主要有线程类、缓存大小、磁盘大小、图片下载与解析、日志方面的配置。
private static ImageLoaderConfiguration config; config = new ImageLoaderConfiguration.Builder(context)// 开始构建 ,图片加载配置 .threadPriority(Thread.NORM_PRIORITY - 2)// 设置线程优先级 .threadPoolSize(3)// 线程池内加载的数量 ;减少配置之中线程池的大小,(.threadPoolSize).推荐1-5; .denyCacheImageMultipleSizesInMemory()// 设置加载的图片有多样的 .tasksProcessingOrder(QueueProcessingType.LIFO)// 图片加载任务顺序 .memoryCache(new WeakMemoryCache())//使用.memoryCache(new WeakMemoryCache()),不要使用.cacheInMemory(); .memoryCacheExtraOptions(480, 800) // 即保存的每个缓存文件的最大长宽 .memoryCacheSizePercentage(60)// 图片内存占应用的60%; .diskCacheFileNameGenerator(new HashCodeFileNameGenerator())//使用HASHCODE对UIL进行加密命名 .diskCacheFileNameGenerator(new Md5FileNameGenerator())// 将保存的时候的URI名称用MD5 加密 .diskCacheSize(50 * 1024 * 1024) // 缓存设置大小为50 Mb .diskCache(new UnlimitedDiskCache(cacheDir))// 自定义缓存路径 .diskCacheFileCount(100) // 缓存的文件数量 .denyCacheImageMultipleSizesInMemory()// 自动缩放 .imageDownloader(new BaseImageDownloader(context, 5 * 1000, 30 * 1000)) // connectTimeout (5 s), readTimeout (30 s)超时时间 .memoryCacheExtraOptions(480, 800)//设置缓存图片时候的宽高最大值,默认为屏幕宽高;保存的每个缓存文件的最大长宽 .defaultDisplayImageOptions(options)// 如果需要打开缓存机制,需要自己builde一个option,可以是DisplayImageOptions.createSimple() .writeDebugLogs() // Remove for release app .build();构建完成(参数可以不用设置全,根据需要来配置)
2.每一个ImageLoader.displayImage(...)都可以使用Display Options。
private DisplayImageOptions options; options = new DisplayImageOptions.Builder()// 开始构建, 显示的图片的各种格式 .resetViewBeforeLoading(true)// 设置图片在下载前是否重置,复位 .cacheInMemory(true)// 开启内存缓存 .cacheOnDisk(true) // 开启硬盘缓存 .displayer(new RoundedBitmapDisplayer(20))// 是否设置为圆角,弧度为多少;避免使用RoundedBitmapDisplayer.他会创建新的ARGB_8888格式的Bitmap对象; .displayer(new FadeInBitmapDisplayer(100))// 是否图片加载好后渐入的动画时间 .displayer(new SimpleBitmapDisplayer())// 正常显示一张图片 .bitmapConfig(Bitmap.Config.RGB_565)// 设置图片的解码类型;使用.bitmapConfig(Bitmap.config.RGB_565)代替ARGB_8888; .considerExifParams(true)// 是否考虑JPEG图像EXIF参数(旋转,翻转) .imageScaleType(ImageScaleType.EXACTLY)// 缩放级别 .imageScaleType(ImageScaleType.IN_SAMPLE_INT)//这两种配置缩放都推荐 .build();// 构建完成(参数可以不用设置全,根据需要来配置) ImageLoader.getInstance().init(config);//初始化完成
3.ImageLoader是显示图片的具体执行类,它有两个具体的方法displayImage loadImage(...)
ImageLoader.getInstance().displayImage(url, imageView, options);