线程方法Bitmap 处理之不要在UI主线程中处理Bitmap

上班之余抽点时间出来写写博文,希望对新接触的朋友有帮助。今天在这里和大家一起学习一下线程方法

    原文链接 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的源资

文章结束给大家分享下程序员的一些笑话语录: 姿势要丰富,经常上百度!

posted @ 2013-04-27 19:56  xinyuyuanm  阅读(923)  评论(0编辑  收藏  举报