Android list加载图片工具类
总体思路
Handler + looper + message
核心类
package com.base.imagechoose.util; import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.os.Handler;
import android.os.Looper;
import android.os.Message;
import android.support.v4.util.LruCache;
import android.util.DisplayMetrics;
import android.view.ViewGroup;
import android.widget.ImageView; import java.lang.reflect.Field;
import java.util.LinkedList;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Semaphore; /**
* 图片加载类
*/
public class ImageLoader { private static ImageLoader mInstance; /**
* 图片缓存
*/
private LruCache<String, Bitmap> mLruCache; private ExecutorService mThreadPool;//线程池, 加载任务 private static final int DAFULT_THREAD_COUNT = 1; //线程数量
/**
* 队列调度方式
*/
private Type mType = Type.FIFO; /**
* 任务队列
*/
private LinkedList<Runnable> mTaskQueue; /**
* 后台轮询线程
*/
private Thread mPoolThread;
private Handler mPoolTheadHandler; //信号量同步mPoolTheadHandler
private Semaphore mSemaphorePoolTheadHandler = new Semaphore(0);
/**
* 用Semaphore来控制一个线程执行完才执行下一个
*/
private Semaphore mSemaphorePool; //用信号量来达到线程Type规则 /**
* UI线程中的Handler
*/
private Handler mUIHandler; //先进先出
public enum Type {
FIFO, FIFI;
} private ImageLoader(int threadCount, Type type) {
init(threadCount, type);
} public static ImageLoader getInstance() {
if (mInstance == null) {
synchronized (ImageLoader.class) {
if (mInstance == null) {
mInstance = new ImageLoader(DAFULT_THREAD_COUNT, Type.FIFO);
}
}
}
return mInstance;
} public static ImageLoader getInstance(int threadCount, Type type) {
if (mInstance == null) {
synchronized (ImageLoader.class) {
if (mInstance == null) {
mInstance = new ImageLoader(threadCount, type);
}
}
}
return mInstance;
} private void init(int threadCount, Type type) {
//后台轮询线程
mPoolThread = new Thread(){ @Override
public void run() {
Looper.prepare();
mPoolTheadHandler = new Handler() {
@Override
public void handleMessage(Message msg) {
//线程池去取出一个任务进行执行
mThreadPool.execute(getTask()); try {
//如果信号量为3,到第4个就会阻塞住
mSemaphorePool.acquire();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
};
//释放一个信号量
mSemaphorePoolTheadHandler.release();
Looper.loop();
}
};
mPoolThread.start(); //获取应用的最大可用内存
int maxMemory = (int) Runtime.getRuntime().maxMemory();
//设置缓存空间
int cacheMemory = maxMemory/8; //初始化缓存
mLruCache = new LruCache<String, Bitmap>(cacheMemory) {
@Override
protected int sizeOf(String key, Bitmap value) {
//返回内存,value每一行的大小*高度
return value.getRowBytes()*value.getHeight();
}
}; //创建线程池
mThreadPool = Executors.newFixedThreadPool(threadCount);
//初始化任务队列
mTaskQueue = new LinkedList<>(); mType = type; mSemaphorePool = new Semaphore(threadCount);
} /**
* 从任务队列取出一个方法
* @return
*/
private Runnable getTask() { if (mType == Type.FIFO) {
return mTaskQueue.removeFirst();
} else {
return mTaskQueue.removeLast();
} } /**
* 根据path为控件加载图片
* @param path 路径
* @param imageView 控件
*/
public void loadImage(final String path, final ImageView imageView) { imageView.setTag(path); if (mUIHandler == null) {
mUIHandler = new Handler() {
@Override
public void handleMessage(Message msg) {
//获取得到的图片并在控件上显示
ImgBeanHolder holder = (ImgBeanHolder) msg.obj; //当tag为传入path时才进行赋值
if (holder.imageView.getTag().equals(holder.path)) {
holder.imageView.setImageBitmap(holder.bm);
}
}
};
} //根据path在缓存中获取图片
Bitmap bm = getBitmapFromLruCache(path); if (bm != null) {
refreashBitmap(path, imageView, bm);
} else {
//没有图片,添加一个任务
addTasks(new Runnable(){
@Override
public void run() {
//加载图片
//图片压缩
//1.获得图片需要显示的大小
ImageSize imageSize = getImageViewSize(imageView);
//2.压缩图片
Bitmap bm = decodeSampleBitmapFromPath(path, imageSize.width, imageSize.height); //3.把图片加入缓存
addBitmapToLruCache(path,bm); refreashBitmap(path, imageView, bm);
//当前任务执行完后释放信号量
mSemaphorePool.release();
}
});
} } /**
* 发送handler显示图片
* @param path
* @param imageView
* @param bm
*/
private void refreashBitmap(String path, ImageView imageView, Bitmap bm) {
Message message = new Message();
ImgBeanHolder imgBeanHolder = new ImgBeanHolder();
imgBeanHolder.bm = bm;
imgBeanHolder.imageView = imageView;
imgBeanHolder.path = path;
message.obj = imgBeanHolder;
mUIHandler.sendMessage(message);
} /**
* 将图片放入缓存
* @param path
* @param bm
*/
private void addBitmapToLruCache(String path, Bitmap bm) {
if(getBitmapFromLruCache(path) == null) {
if (bm != null) {
mLruCache.put(path, bm);
}
} } /**
* 添加任务
* @param runnable
*/
private synchronized void addTasks(Runnable runnable) {
//将任务放入任务队列中
mTaskQueue.add(runnable);
//发送一个通知去通知后台轮询执行任务
try {
if (mPoolTheadHandler == null)
mSemaphorePoolTheadHandler.acquire();
} catch (InterruptedException e) {
e.printStackTrace();
}
mPoolTheadHandler.sendEmptyMessage(0x110);
} /**
* 根据path在缓存中获取图片
* @param key
* @return
*/
private Bitmap getBitmapFromLruCache(String key) { return mLruCache.get(key);
} /**
* 根据imageview获取图片需要压缩的宽和高
* @param imageView
*/
private ImageSize getImageViewSize(ImageView imageView) {
ImageSize imageSize = new ImageSize(); DisplayMetrics displayMetrics = imageView.getContext().getResources().getDisplayMetrics();
ViewGroup.LayoutParams lp = imageView.getLayoutParams(); int width = imageView.getWidth(); if (width <= 0) {
width = lp.width; //获取imageview在layout中声明的亮度
} if (width <= 0) {
width = getImageViewFileValue(imageView, "mMaxWidth"); //检查最大值
} if (width <= 0) {
width = displayMetrics.widthPixels;
} int height = imageView.getHeight(); if (height <= 0) {
height = lp.height; //获取imageview在layout中声明的亮度
} if (height <= 0) {
height = getImageViewFileValue(imageView, "mMaxHeight"); //检查最大值 //检查最大值
} if (height <= 0) {
height = displayMetrics.heightPixels;
} imageSize.width = width;
imageSize.height = height; return imageSize;
} private static int getImageViewFileValue(Object object,String fileName) {
int value = 0; //得到当前ImageView的字段集合
try {
Field field = ImageView.class.getDeclaredField(fileName);
field.setAccessible(true); int fieldValue = field.getInt(object); if (fieldValue > 0 && fieldValue < Integer.MAX_VALUE) {
value = fieldValue;
}
} catch (Exception e) {
e.printStackTrace();
} return value;
} /**
* 根据图片需要的宽高对图片进行压缩
* @param path
* @param width
* @param height
* @return
*/
private Bitmap decodeSampleBitmapFromPath(String path, int width, int height) { BitmapFactory.Options options = new BitmapFactory.Options();
//不加入内存
options.inJustDecodeBounds = true;
//获取实际的options
BitmapFactory.decodeFile(path, options); //用实际的options和需要显示的宽高算出inSampleSize
options.inSampleSize = caculateInSampleSize(options, width, height); //使用获取的inSampleSize再次解析图片
//放入内存
options.inJustDecodeBounds = false; Bitmap bitmap = BitmapFactory.decodeFile(path, options); return bitmap;
} /**
* 用实际的options和需要显示的宽高算出inSampleSize
* @param options
* @param reqWidth
* @param reqHeight
* @return
*/
private int caculateInSampleSize(BitmapFactory.Options options, int reqWidth, int reqHeight) { int width = options.outWidth;
int height = options.outHeight; int inSampleSize = 1; if (width > reqHeight || height > reqHeight) {
int widthRadio = Math.round(width*1.0f/reqWidth);
int heightRadio = Math.round(height*1.0f/reqHeight); inSampleSize = Math.max(widthRadio, heightRadio);
} return inSampleSize;
} private class ImageSize{
int width;
int height;
} private class ImgBeanHolder {
Bitmap bm;
String path;
ImageView imageView;
} }
Android list加载图片工具类的更多相关文章
- android异步加载图片并缓存到本地实现方法
图片过多造成内存溢出,这个是最不容易解决的,要想一些好的缓存策略,比如大图片使用LRU缓存策略或懒加载缓存策略.今天首先介绍一下本地缓存图片 在android项目中访问网络图片是非常普遍性的事 ...
- android listview 加载图片错乱(错位)
写道 今天晚上一个朋友介绍我看了一篇文章,也是解决android中listview在加载图片错位的问题,看了之后,感觉写的很好,自己也遇到这个问题,但是又不知道从何下手,看到这篇文章后,我的问题 ...
- Android Glide加载图片时转换为圆形、圆角、毛玻璃等图片效果
Android Glide加载图片时转换为圆形.圆角.毛玻璃等图片效果 附录1简单介绍了Android开源的图片加载框架.在实际的开发中,虽然Glide解决了快速加载图片的问题,但还有一个问题悬 ...
- 演化理解 Android 异步加载图片(转)
演化理解 Android 异步加载图片(转)http://www.cnblogs.com/CJzhang/archive/2011/10/20/2218474.html
- Android 异步加载图片,使用LruCache和SD卡或手机缓存,效果非常的流畅
Android 高手进阶(21) 版权声明:本文为博主原创文章,未经博主允许不得转载. 转载请注明出处http://blog.csdn.net/xiaanming/article/details ...
- android 网络加载图片,对图片资源进行优化,并且实现内存双缓存 + 磁盘缓存
经常会用到 网络文件 比如查看大图片数据 资源优化的问题,当然用开源的项目 Android-Universal-Image-Loader 或者 ignition 都是个很好的选择. 在这里把原来 ...
- 实例演示Android异步加载图片
本文给大家演示异步加载图片的分析过程.让大家了解异步加载图片的好处,以及如何更新UI.首先给出main.xml布局文件:简单来说就是 LinearLayout 布局,其下放了2个TextView和5个 ...
- 实例演示Android异步加载图片(转)
本文给大家演示异步加载图片的分析过程.让大家了解异步加载图片的好处,以及如何更新UI.首先给出main.xml布局文件:简单来说就是 LinearLayout 布局,其下放了2个TextView和5个 ...
- [Android]异步加载图片,内存缓存,文件缓存,imageview显示图片时增加淡入淡出动画
以下内容为原创,欢迎转载,转载请注明 来自天天博客:http://www.cnblogs.com/tiantianbyconan/p/3574131.html 这个可以实现ImageView异步加载 ...
随机推荐
- (转载)Windows无法安装到GPT分区形式磁盘解决办法
之前使用的是windows7 + ubuntu18.04双系统,硬盘分区采用的是GPT格式.重装windows系统时,提示“windows无法安装到这个磁盘.选中的磁盘采用GPT分区形式”,导致安装失 ...
- JavaScript中的真和假,==和===, 不等
咋JS中,下面这些值表示 “假”: "" (empty string) 0,-0,NaN (invalid number) null, undefined false 除了上面这些 ...
- hdu1067
#include<iostream> #include<cstdio> #include<cstring> #include<cmath> #inclu ...
- Linux ifconfig-etho文件参数详解
1.ifcfg-eth0文件参数详解 DEVICE 网卡名称/网络接口的名称BOOTPROTO 系统启动地址协议 常用参数: none:不使用启动地址协议,none禁止DHCP bootp:BOOTP ...
- 学习笔记:首次进行JUnit+Ant构建自动的单元测试(二)
关键字:JUnit,Ant,单元测试 终于把JUnit+Ant构建单元测试的大概了解了,其实我实践的过程是对了,只是指导博客(看到这里不懂请看我上一篇博客)本身的错误“成功”把我带入“坑”,有时候网友 ...
- es6- Generator函数实现长轮询
1.Generator 函数是 ES6 提供的一种异步编程解决方案,语法行为与传统函数完全不同. 语法上,首先可以把它理解成,Generator 函数是一个状态机,封装了多个内部状态.形式上,Gene ...
- 《OD学hadoop》20160910某旅游网项目实战
一.event事件分析 叶子节点只计算一次 父节点的触发次数由子节点的数量节点 事件流:是由业务人员定义的一系列的具有前后顺序的事件构成的用户操作行为,至少包括两个事件以上. 目标:以事件流为单位,分 ...
- 洛谷P1378 油滴扩展
P1378 油滴扩展 题目描述 在一个长方形框子里,最多有N(0≤N≤6)个相异的点,在其中任何一个点上放一个很小的油滴,那么这个油滴会一直扩展,直到接触到其他油滴或者框子的边界.必须等一个油滴扩展完 ...
- SPGroup 和SPUser的常用操作
http://www.cnblogs.com/gzh4455/archive/2012/03/26/2417854.html private bool RemoveUserFromGroup(stri ...
- 如何理解javascript中的同步和异步
javascript语言是一门“单线程”的语言,不像java语言,类继承Thread再来个thread.start就可以开辟一个线程,所以,javascript就像一条流水线,仅仅是一条流水线而已,要 ...