ListView异步加载图片是十分有用的办法,但凡是要经过网络获取图片资源一般运用这种办法比较好,用户体会好,不必让用户等候下去,下面就说完结办法,先贴上主办法的代码
public class AsyncImageLoader {
private HashMap
public AsyncImageLoader() {
imageCache = new HashMap
}
public Drawable loadDrawable(final String imageUrl, final ImageCallback imageCallback) {
if (imageCache.containsKey(imageUrl)) {
SoftReference
Drawable drawable = softReference.get();
if (drawable != null) {
return drawable;
}
}
final Handler handler = new Handler() {
public void handleMessage(Message message) {
imageCallback.imageLoaded((Drawable) message.obj, imageUrl);
}
};
new Thread() {
@Override
public void run() {
Drawable drawable = loadImageFromUrl(imageUrl);
imageCache.put(imageUrl, new SoftReference
Message message = handler.obtainMessage(0, drawable);
handler.sendMessage(message);
}
}.start();
return null;
}
public static Drawable loadImageFromUrl(String url) {
URL m;
InputStream i = null;
try {
m = new URL(url);
i = (InputStream) m.getContent();
} catch (MalformedURLException e1) {
e1.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
Drawable d = Drawable.createFromStream(i, src);
return d;
}
public interface ImageCallback {
public void imageLoaded(Drawable imageDrawable, String imageUrl);
}
}
完结办法:经过传入图片的网络地址和一个完结ImageCallback行为的目标,当imageCache存在这个图片时分,回来这个图片,当imageCache没有这个图片时,实例一个异步线程来下载图片并一同回来为null,最终在图片下载完结的时分,调用imageLoaded办法。
现说说这个类规划的长处吧:1.采用了战略形式;2.运用SoftReference关键字
先说说战略形式,程序里把每次下载图片完结后所进行的操作封装成一个ImageCallback抽象类,使体系更灵敏,并易于扩展。
在Java中内存办理,引证分为四大类,强引证HardReference、弱引证WeakReference、软引证SoftReference和虚引证PhantomReference。它们的差异也很明显,HardReference目标是即便虚拟机内存吃紧抛出OOM也不会导致这一引证的目标被收回,而WeakReference等更适合于一些数量不多,但体积略微巨大的目标,在这四个引证中,它是最简单被废物收回的,而咱们关于显现相似Android Market中每个运用的App Icon时能够考虑运用SoftReference来处理内存不至于快速收回,一同当内存缺少面对Java VM溃散抛出OOM前时,软引证将会强制收回内存,最终的虚引证一般没有实际意义,只是调查GC的活动状况,关于测验比较有用一同有必要和ReferenceQueue一同运用。关于一组数据,咱们能够经过HashMap的办法来增加一组SoftReference目标来暂时保存一些数据,一同关于需求重复经过网络获取的不常常改动的内容,能够经过本地的文件体系或数据库来存储缓存。
最终一句话说的很对,事实上大多数状况也是如此。
在说说它的用法吧,一般它作为一个adapter的一个变量如:
class BookAdapter extends ArrayAdapter
AsyncImageLoader asyncImageLoader;
Context mContext;
BookAdapter(Context context,List
super(context, 0, data);
asyncImageLoader = new AsyncImageLoader();
mContext = context;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
ViewCache holder ;
if(convertView==null){
LayoutInflater inflate = (LayoutInflater)mContext.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
convertView = inflate.inflate(com.slider.cn.R.layout.list_item , null);
holder = new ViewCache();
holder.icon = (ImageView)convertView.findViewById(com.slider.cn.R.id.note_icon);
holder.name = (TextView)convertView.findViewById(com.slider.cn.R.id.note_name);
holder.date = (TextView)convertView.findViewById(com.slider.cn.R.id.note_date);
convertView.setTag(holder);
}else{
holder = (ViewCache)convertView.getTag();
}
final BookInfo bookInfo = getItem(position);
holder.name.setText(bookInfo.getName().toString());
holder.date.setText(bookInfo.getInfo());
holder.icon.setTag(bookInfo.getUri());
//
Drawable drawable = asyncImageLoader.loadDrawable(bookInfo.getUri(), new ImageCallback() {
@Override
public void imageLoaded(Drawable imageDrawable, String imageUrl) {
ImageView imageViewByTag = (ImageView) BookListView.this.findViewWithTag(bookInfo.getUri());
if (imageViewByTag!=null) {
imageViewByTag.setImageDrawable(imageDrawable);
}else {
//load image failed from Internet
}
}
});
if(drawable==null){
holder.icon.setImageDrawable(drawable_waiting);