上班之余抽点时间出来写写博文,希望对新接触的朋友有帮助。今天在这里和大家一起学习一下线程方法
原文链接 http://developer.android.com/intl/zh-CN/training/displaying-bitmaps/process-bitmap.html
在用使BitmapFactory.decode*剖析图片时,最好不要在UI线主程中处置,因为图片的源来是未知的,有多是从硬盘取读的,也有多是是网络的图片源资,这时在剖析图片时,会有一些不可控的要素,如(网速较慢等),如果在UI线主中处置,就会有可能block线主程,从而致导用应无应相(ANR),会形成很欠好的用户体验。Android本身供提很多的方法,在非UI线程中处置一些比拟耗时的作操,如Handler,AsyncTask等,面下是用使AsyncTask的一种方法。
用使AsyncTask
AsyncTask供提了一种很简单的式方,在后台线程处置庞杂的作操,处置成完后之,会将处置后之的结果返回到UI线主程,详细的现实入下:
class BitmapWorkerTask extends AsyncTask<Integer, Void, Bitmap> {
private final WeakReference<ImageView> imageViewReference;
private int data = 0;
public BitmapWorkerTask(ImageView imageView) {
// Use a WeakReference to ensure the ImageView can be garbage collected
imageViewReference = new WeakReference<ImageView>(imageView);
}
// Decode image in background.
@Override
protected Bitmap doInBackground(Integer... params) {
data = params[0];
return decodeSampledBitmapFromResource(getResources(), data, 100, 100));
}
// Once complete, see if ImageView is still around and set bitmap.
@Override
protected void onPostExecute(Bitmap bitmap) {
if (imageViewReference != null && bitmap != null) {
final ImageView imageView = imageViewReference.get();
if (imageView != null) {
imageView.setImageBitmap(bitmap);
}
}
}
}
这个例子是ImageView在置设大的图片的一个用应 WeakReference<ImageView> imageViewReference 用使一个软用引,保障ImageView可以被回收, 要主的耗时的作操在doInBackground(Integer... params) 方法中施实,处置成完后之的返回的结果在onPostExecute方法中用使,onPostExecute是在线主程运行的。 方法decodeSampledBitmapFromResource 的详细现实请参考 “Bitmap处置之畅流的加载大的Bitmap”
际实的用应:
public void loadBitmap(int resId, ImageView imageView) {
BitmapWorkerTask task = new BitmapWorkerTask(imageView);
task.execute(resId);
}
两个参数分别是 resId是通过R.drawable 得获的源资图片,imageView是 要表现的图片的ImageView
处置并发问题
对于一般的View如ListViwe和GridView在结合上述的AsyncTask加载源资时,会致导其他的问题。为了更好的利用存内源资,这些Viwe在滑动时,会对子View做源资回收,如果子View用使AsyncTask加载源资时,就不能保障子View可以实时的回收源资,也就是说开始一步加载的序顺和成完的序顺会不一致。 关于多线程处置的问题可以参考 “Multithreading for Performance ”的一篇Blog 可以建创一个专有的BitmapDrawable,在AsyncTask加载图片成完之前,可以用使一个empty(预览图)bitmap来表现
static class AsyncDrawable extends BitmapDrawable {
private final WeakReference<BitmapWorkerTask> bitmapWorkerTaskReference;
public AsyncDrawable(Resources res, Bitmap bitmap,
BitmapWorkerTask bitmapWorkerTask) {
super(res, bitmap);
bitmapWorkerTaskReference =
new WeakReference<BitmapWorkerTask>(bitmapWorkerTask);
}
public BitmapWorkerTask getBitmapWorkerTask() {
return bitmapWorkerTaskReference.get();
}
}
在BitmapWorkerTask 执行之前,可以先用使一个预览图表现(或者一张空的图片)
每日一道理
翻开早已发黄的页张,试着寻找过去所留下的点点滴滴的足迹。多年前的好友似乎现在看来已变得陌生,匆忙之间,让这维持了多年的友谊变淡,找不出什么亲切感,只是偶尔遇上,淡淡地微笑,如今也只能在这发黄的页张中找寻过去的那些让人难忘的,至少我可以握住这仅剩下一段的“丝线头”……
翻开早已发黄的页张,试着寻找过去所留下的点点滴滴的足迹。多年前的好友似乎现在看来已变得陌生,匆忙之间,让这维持了多年的友谊变淡,找不出什么亲切感,只是偶尔遇上,淡淡地微笑,如今也只能在这发黄的页张中找寻过去的那些让人难忘的,至少我可以握住这仅剩下一段的“丝线头”……
public void loadBitmap(int resId, ImageView imageView) {
if (cancelPotentialWork(resId, imageView)) {
final BitmapWorkerTask task = new BitmapWorkerTask(imageView);
final AsyncDrawable asyncDrawable =
new AsyncDrawable(getResources(), mPlaceHolderBitmap, task);
imageView.setImageDrawable(asyncDrawable);
task.execute(resId);
}
}
mPlaceHolderBitmap可是以一个empty_photo.png的图片
cancelPotentialWork 方法是检查是不是有另外一个running task和ImageView绑定,如果是,则前一个task会执行cancel()
public static boolean cancelPotentialWork(int data, ImageView imageView) {
final BitmapWorkerTask bitmapWorkerTask = getBitmapWorkerTask(imageView);
if (bitmapWorkerTask != null) {
final int bitmapData = bitmapWorkerTask.data;
if (bitmapData != data) {
// Cancel previous task
bitmapWorkerTask.cancel(true);
} else {
// The same work is already in progress
return false;
}
}
// No task associated with the ImageView, or an existing task was cancelled
return true;
}
如果返回false,说明已有一个工作者线程在运行 getBitmapWorkerTask方法是得获以后ImageView相干的工作者线程
private static BitmapWorkerTask getBitmapWorkerTask(ImageView imageView) {
if (imageView != null) {
final Drawable drawable = imageView.getDrawable();
if (drawable instanceof AsyncDrawable) {
final AsyncDrawable asyncDrawable = (AsyncDrawable) drawable;
return asyncDrawable.getBitmapWorkerTask();
}
}
return null;
}
新更BitmapWorkerTask类
class BitmapWorkerTask extends AsyncTask<Integer, Void, Bitmap> {
...
@Override
protected void onPostExecute(Bitmap bitmap) {
if (isCancelled()) {
bitmap = null;
}
if (imageViewReference != null && bitmap != null) {
final ImageView imageView = imageViewReference.get();
final BitmapWorkerTask bitmapWorkerTask =
getBitmapWorkerTask(imageView);
if (this == bitmapWorkerTask && imageView != null) {
imageView.setImageBitmap(bitmap);
}
}
}
}
加入了断判条件,断判以后的线程是不是已被cancel,ImageView的线程是不是和以后的线程是不是是同一个线程
用应场景
如在ListView或者GridView的getView()方法中,可以点用loadBitmap()方法来加载子view的源资
文章结束给大家分享下程序员的一些笑话语录: 姿势要丰富,经常上百度!