AsyncTask是执行后台线程的最简单方式,但它不适用于那些重复且长时间运行的任务。

 

1. Looper

Android中,线程拥有一个消息队列(message queue),使用消息队列的线程叫做消息循环(message loop)。消息循环会循环检查队列上是否有新消息。

消息循环由线程和looper组成,Looper对象管理着线程的消息队列。

主线程就是个消息循环,因此也拥有Looper,主线程的所有工作都是由其looper完成的,looper不断的从消息队列中抓去消息,然后完成消息指定的任务。

 

2. Message

消息是Message类的一个实例,它有好几个实例变量,其中有三个需要在实现时定义。

1.What:用户定义的int型消息代码,用来描述消息。

2.obj:随消息发送的用户指定对象。

3.target: 处理消息的Handler。

 

3. Handler

Message的目标(target)是Handler类的一个实例,Handler可看作message handler的简称,创建Message时,它会自动与一个handler相关联,message待处理时,Handler对象负责触发消息处理事件、

Handler不仅仅是处理Message的目标,也是创建和发布Message的接口。

 

4. 关系

Looper拥有Message收件箱,所以Message必须在Looper上发布或处理。为与Looper协同工作,Handler总是引用着它。

一个Handler仅与一个Looper相关联,一个Message也仅于一个目标Handler(也称作Message目标)相关联。Looper拥有整个Message队列,多个Message可以引用同一目标Handler。

多个Handler也可与一个Looper相关联,这意味着一个Handler的Message可能与另一个Handler的Message存放在同一消息队列中。

 

5. 使用Handler 

通常不需要手动设置消息的目标Handler。创建信息时,调用Handler.obtainMessage()方法。当传入其他消息字段给他时,该方法会自动设置目标给Handler对象。

为避免创建新的Message对象,Handler.obtainMessage()方法会从公共循环吃获取消息。  

一旦取得Message,就可以调用sendToTarget()方法将其发送给它的Handler,然后Handler会将这个Message放置在Looper消息队列的尾部。

Looper取得消息队列中的特定消息后,会将它发送给消息目标去处理。消息一般是在目标的Handler.handleMessage()实现方法中进行处理。

 

6. HandlerThread

HandlerThread类帮我们完成了建立looper的过程,只要继承它就能省去一些工作。

public class ThumbnailDownloader<T> extends HandlerThread {
    private static final String TAG = "ThumbnailDownloader";
    private static final int MESSAGE_DOWNLOAD = 0; //标识下载请求
    private Boolean mHasQuit = false;
    private Handler mRequestHandler; //存储对Handler的引用,这个Handler负责在ThumbnailDownloader后台线程上管理下载请求消息队列。这个Handler也负责从消息队列里取出并处理下载请求消息。

    //onLooperPrepared()在Looper首次检查消息队列之前调用的。
    @Override
    protected void onLooperPrepared(){
        mRequestHandler = new Handler(){
            @Override
            public void handleMessage(Message msg){  //队列中的下载消息取出并可以处理时,就会触发调用Handler.handleMessage()方法。
                //处理操作
            }
        };
    }

    public void queueThumbnail(T target, String url) {
 //当传入其他消息字段给它时,该方法会自动设置目标给Handler对象(obtainMessage)
            //sendToTarget()方法将Message发送给它的Handler,然后Handler会将这个Message放置在Looper消息队列的尾部。
               mRequestHandler.obtainMessage(MESSAGE_DOWNLOAD,target).sendToTarget();
    }
    public void clearQueue(){
        mRequestHandler.removeMessages(MESSAGE_DOWNLOAD);
    }
    
}

 

主线程中这样调用:

mThumbnailDownloader = new ThumbnailDownloader<>(responseHandler);
mThumbnailDownloader.start();
mThumbnailDownloader.getLooper(); //在start()方法之后调用getLooper()方法是一种保证线程就绪的处理方式。可以避免潜在竞争。
// 在需要的时候调用
mThumbnailDownloader.queueThumbnail(holder, url);

 

7.线程交互

主线程现在能够适时调用这个线程的方法,用于下载图片了。但是还存在一个问题,那就是下载线程下载完一个任务以后如何更新视图呢?我们知道 UI 只能在主线程里更新,所以我们采用在主线程里声明一个 Handler,传递给下载线程,让下载线程在下载完成后在主线程执行更新操作。因为不能直接引用主线程的方法,故而在这里用到了回调。

7.1下载线程中:

// ThumbnailDownloader,也就是下载线程中

// 成员声明
private Handler mResponseHandler;
private ThumbnailDowloadListener<T> mThumbnailDownloadListener;

// 回调接口
public interface ThumbnailDowloadListener<T> {
/*
         * 图片下载完成,可以交给UI去显示时,接口中的方法就会被调用。
         * 会使用这个方法把处理已下载图片的任务代理给另一个类(PhotoGalleryFragment),这样ThumbnailDownloader就可以把下载结果传给其他视图对象。
         */
    void onThumbnailDownloaded(T target, Bitmap thumbnail);
}

public void setThumbnailDownloaderListener(ThumbnailDowloadListener<T> listener) {
    mThumbnailDownloadListener = listener;
}

// 通过构造函数传递主线程的 Handler
public ThumbnailDowloader(Handler responseHandler) {
    super(TAG);
    mResponseHandler = responseHandler;
}

这样,主线程通过调用这些方法,就能够让下载线程获取到主线程的 Handler 和回调接口实例。

7.2主线程中

// 成员声明
private ThumbnailDowloader<PhotoHolder> mThumbnailDownloader;

// 传递实例给下载线程
// 这个 Handler 在主线程中建立,所以是和主线程 Looper 相关联的
Handler responseHandler = new Handler(); 
mThumbnailDownloader = new ThumbnailDowloader<>(responseHandler);
mThumbnailDownloader.setThumbnailDownloaderListener(
    new ThumbnailDowloader.ThumbnailDowloadListener<PhotoHolder>() {
        @Override
       public void onThumbnailDownloaded(PhotoHolder target, Bitmap thumbnail) {
            Drawable drawable = new BitmapDrawable(getResources(), thumbnail);
            target.bindDrawable(drawable);
        }
    }
);

8.线程交互

现在,通过 mResponseHandler,下载线程能够访问与主线程 Looper 绑定的 Handler。同时,还有 ThumbnailDownloadListener 使用返回的 Bitmap 执行 UI 更新操作。具体来说, 就是通过 onThumbnailDownloaded 实现,使用新下载的 Bitmap 来设置 PhotoHolder 的 Drawable。 
和在下载线程上把下载图片的请求放入消息队列类似,我们也可以返回定制 Message 给主线程,要求显示已下载图片。不过,这需要另一个 Handler 子类,以及一个 handleMessage(…) 覆盖方法。方便起见,我们转而使用另一个方便的 Handler 方法——post(Runnable)。

mResponseHandler.post(new Runnable() {
    @Override
    public void run() {
        if (mRequestMap.get(target) != url ||
                mHasQuit) {
            return;
        }

        mRequestMap.remove(target);
        mThumbnailDownloadListener.onThumbnailDownloaded(target, bitmap);
    }
});

在这里,新建的 Runnable 对象会被当成 Message 的回调方法,直接执行 run() 方法,所以相当于发送一个消息,里面写明了怎么做,而不是把对象和消息类型发给 Handler,让 Handler 决定怎么做。