android异步加载图片类(续)-universal-image-loader详解

2024-06-06 11:58

本文主要是介绍android异步加载图片类(续)-universal-image-loader详解,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

之前写过一篇android异步加载图片类 ,后来接触了一个开源项目universal-image-loader,听说淘宝也是用这玩意

发现自己写的那个异步加载类太简单了,虽然功能是实现了,但是很多优化的问题都没有解决

比如:

同一个ui加载同一张图,会出现只加载一张,其他的加载不了

加载多图的时候会有oom等问题


现在来说说universal-image-loader

特点:

多线程的图像加载
的可能性的宽调谐对ImageLoader的配置(线程池的大小,HTTP选项,内存和光盘高速缓存,显示图像,以及其他)
的图像的可能性中的缓存存储器和/或设备的文件器系统(或SD卡)
可以“听”加载过程中
可自定义每个显示的图像调用分隔的选项
Widget支持
Android 1.5以上支持


使用方法:(我自己封装了一个类ImgConfig使用,方便一些)

只需两步即可加载网络图片:

一、初始化

ImgConfig.initImageLoader();

二、

ImgConfig.showUserSImg(imgUrl, imageview); //图片的url,要显示的view


以下是我自己写的ImgConfig,仅供参考:

public class ImgConfig extends ImageLoader {
private static DisplayImageOptions options_corner;
private static DisplayImageOptions options_square;
private static AnimateFirstDisplayListener animateFirstDisplayListener = new AnimateFirstDisplayListener();
/**
* @param url   服务器的文件名
* @param imageView
*            显示方形图片 S for Square
*/
public static void showUserSImg(String url, ImageView imageView) {
ImageLoader.getInstance().displayImage(url,
imageView, options_square, animateFirstDisplayListener);
}
/**
* @param url   服务器的文件名
* @param imageView
*            圆角 C for Corner
*/
public static void showUserCImg(String url, ImageView imageView) {
ImageLoader.getInstance().displayImage(url,
imageView, options_corner, animateFirstDisplayListener);
}
/**
* 初始化图片读取方式
*/
public static void initImageLoader() {
DisplayImageOptions options_corner = new DisplayImageOptions.Builder()
.showImageOnLoading(ImgHandler.ToCircular(R.drawable.defult_head))
// 加载中
.showImageForEmptyUri(ImgHandler.ToCircular(R.drawable.defult_head))
// 空uri
.showImageOnFail(ImgHandler.ToCircular(R.drawable.defult_head))
// 失败时
.cacheInMemory(true)
// 设置下载的图片是否缓存在内存中
.cacheOnDisc(true)
// 设置下载的图片是否缓存在SD卡中
.considerExifParams(true)
.displayer(new RoundedBitmapDisplayer(10)) // 展现方式:圆角
.resetViewBeforeLoading(true)
.imageScaleType(ImageScaleType.EXACTLY)
// new FadeInBitmapDisplayer(300) 渐现
.build();
options_square = new DisplayImageOptions.Builder()
.showImageOnLoading(R.drawable.defult_head)
.showImageForEmptyUri(R.drawable.defult_head)
.showImageOnFail(R.drawable.defult_head)
.cacheInMemory(true)
.cacheOnDisc(true)
.considerExifParams(true)
.displayer(new FadeInBitmapDisplayer(100)) // 展现方式:渐现
.resetViewBeforeLoading(true)
.imageScaleType(ImageScaleType.EXACTLY)
.build();
// This configuration tuning is custom. You can tune every option, you
// may tune some of them,
// or you can create default configuration by
// ImageLoaderConfiguration.createDefault(this);
// method.
ImageLoaderConfiguration config = new ImageLoaderConfiguration.Builder(
ContextUtil.getInstance())
.threadPriority(Thread.NORM_PRIORITY)  //线程池的数量
.denyCacheImageMultipleSizesInMemory() // 不同大小图片只有一个缓存,默认多个
.tasksProcessingOrder(QueueProcessingType.LIFO)
// 设置图片下载和显示的工作队列排序
.discCache(new LimitedAgeDiscCache(new File(Constant.SAVE_IMG_PATH),
new Md5FileNameGenerator(), 7 * 24 * 60 * 60)) // 7天自动清除,按秒算
// .writeDebugLogs() // Remove for release app
.imageDownloader(     //或许你的服务器有特定的加载图的方式,在这里实现
new BaseImageDownloader(ContextUtil.getInstance()) {
@Override
public InputStream getStream(String imageUri,
Object extra) throws IOException {
return super.getStream(imageUri, extra);
}
@Override
protected InputStream getStreamFromNetwork(
String imageUri, Object extra)
throws IOException {
HttpURLConnection conn = createConnection(
imageUri, extra);
int redirectCount = 0;
while (conn.getResponseCode() / 100 == 3
&& redirectCount < MAX_REDIRECT_COUNT) {
conn = createConnection(
conn.getHeaderField("Location"),
extra);
redirectCount++;
}
InputStream imageStream = null;
try {
imageStream = conn.getInputStream();
} catch (IOException e) {
// Read all data to allow reuse connection
// (http://bit.ly/1ad35PY)
IoUtils.readAndCloseStream(conn
.getErrorStream());
}
return new ContentLengthInputStream(
new BufferedInputStream(imageStream,
BUFFER_SIZE), conn
.getContentLength());
}
}).build();
// Initialize ImageLoader with configuration.  初始化
ImageLoader.getInstance().init(config);
}
/**
* @author Administrator 监听读取完图片
*/
private static class AnimateFirstDisplayListener extends
SimpleImageLoadingListener {
// 放到内存
static final List<String> displayedImages = Collections
.synchronizedList(new LinkedList<String>());
@Override
public void onLoadingComplete(String imageUri, View view,
Bitmap loadedImage) {
if (loadedImage != null) {
ImageView imageView = (ImageView) view;
boolean firstDisplay = !displayedImages.contains(imageUri);
if (firstDisplay) {
FadeInBitmapDisplayer.animate(imageView, 500);
displayedImages.add(imageUri);
}
}
}
}
}

特别提示以下 .disCache() 和 .imageDownloader

因为我不想要自动帮我加密文件,所以我.disCache() 里面的加密方法 new Md5FileNameGenerator()

换成了

new FileNameGenerator() {
@Override
public String generate(String imageUri) {
return FileUtil.getFileName(imageUri);
}
}

服务器有自己的加载方式,我把加载 .imageDownloader的加载图片方法换了

new BaseImageDownloader(ContextUtil.getInstance()) {
@Override
public InputStream getStream(String imageUri,
Object extra) throws IOException {
return super.getStream(imageUri, extra);
}
@Override
protected InputStream getStreamFromNetwork(
String imageUri, Object extra)
throws IOException {
HttpURLConnection conn = createConnection(
imageUri, extra);
int redirectCount = 0;
while (conn.getResponseCode() / 100 == 3
&& redirectCount < MAX_REDIRECT_COUNT) {
conn = createConnection(
conn.getHeaderField("Location"),
extra);
redirectCount++;
}
InputStream imageStream = null;
try {
imageStream = conn.getInputStream();
} catch (IOException e) {
// Read all data to allow reuse connection
// (http://bit.ly/1ad35PY)
IoUtils.readAndCloseStream(conn
.getErrorStream());
}
return new ContentLengthInputStream(
new BufferedInputStream(imageStream,
BUFFER_SIZE), conn
.getContentLength());
}
}


总结: universal-image-loader 这个开源项目很好用,

研究了一下源码,有很多启发,扩展性很好,值得学习

官方提供的例子: universal-image-loader例子


这篇关于android异步加载图片类(续)-universal-image-loader详解的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

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

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

一文详解SpringBoot中控制器的动态注册与卸载

《一文详解SpringBoot中控制器的动态注册与卸载》在项目开发中,通过动态注册和卸载控制器功能,可以根据业务场景和项目需要实现功能的动态增加、删除,提高系统的灵活性和可扩展性,下面我们就来看看Sp... 目录项目结构1. 创建 Spring Boot 启动类2. 创建一个测试控制器3. 创建动态控制器注

C#读写文本文件的多种方式详解

《C#读写文本文件的多种方式详解》这篇文章主要为大家详细介绍了C#中各种常用的文件读写方式,包括文本文件,二进制文件、CSV文件、JSON文件等,有需要的小伙伴可以参考一下... 目录一、文本文件读写1. 使用 File 类的静态方法2. 使用 StreamReader 和 StreamWriter二、二进

Conda与Python venv虚拟环境的区别与使用方法详解

《Conda与Pythonvenv虚拟环境的区别与使用方法详解》随着Python社区的成长,虚拟环境的概念和技术也在不断发展,:本文主要介绍Conda与Pythonvenv虚拟环境的区别与使用... 目录前言一、Conda 与 python venv 的核心区别1. Conda 的特点2. Python v

Spring Boot中WebSocket常用使用方法详解

《SpringBoot中WebSocket常用使用方法详解》本文从WebSocket的基础概念出发,详细介绍了SpringBoot集成WebSocket的步骤,并重点讲解了常用的使用方法,包括简单消... 目录一、WebSocket基础概念1.1 什么是WebSocket1.2 WebSocket与HTTP

java中反射Reflection的4个作用详解

《java中反射Reflection的4个作用详解》反射Reflection是Java等编程语言中的一个重要特性,它允许程序在运行时进行自我检查和对内部成员(如字段、方法、类等)的操作,本文将详细介绍... 目录作用1、在运行时判断任意一个对象所属的类作用2、在运行时构造任意一个类的对象作用3、在运行时判断

MySQL 中的 CAST 函数详解及常见用法

《MySQL中的CAST函数详解及常见用法》CAST函数是MySQL中用于数据类型转换的重要函数,它允许你将一个值从一种数据类型转换为另一种数据类型,本文给大家介绍MySQL中的CAST... 目录mysql 中的 CAST 函数详解一、基本语法二、支持的数据类型三、常见用法示例1. 字符串转数字2. 数字

SpringBoot中SM2公钥加密、私钥解密的实现示例详解

《SpringBoot中SM2公钥加密、私钥解密的实现示例详解》本文介绍了如何在SpringBoot项目中实现SM2公钥加密和私钥解密的功能,通过使用Hutool库和BouncyCastle依赖,简化... 目录一、前言1、加密信息(示例)2、加密结果(示例)二、实现代码1、yml文件配置2、创建SM2工具

MyBatis-Plus 中 nested() 与 and() 方法详解(最佳实践场景)

《MyBatis-Plus中nested()与and()方法详解(最佳实践场景)》在MyBatis-Plus的条件构造器中,nested()和and()都是用于构建复杂查询条件的关键方法,但... 目录MyBATis-Plus 中nested()与and()方法详解一、核心区别对比二、方法详解1.and()

Spring IoC 容器的使用详解(最新整理)

《SpringIoC容器的使用详解(最新整理)》文章介绍了Spring框架中的应用分层思想与IoC容器原理,通过分层解耦业务逻辑、数据访问等模块,IoC容器利用@Component注解管理Bean... 目录1. 应用分层2. IoC 的介绍3. IoC 容器的使用3.1. bean 的存储3.2. 方法注