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 决定怎么做。

安卓权威编程指南-笔记(第24章 Looper Handler 和 HandlerThread)的更多相关文章

  1. 安卓权威编程指南 -笔记(19章 使用SoundPool播放音频)

    针对BeatBox应用,可以使用SoundPool这个特别定制的实用工具. SoundPool能加载一批声音资源到内存中,并支持同时播放多个音频文件.因此所以,就算用户兴奋起来,狂按按钮播放全部音频, ...

  2. 安卓权威编程指南 -笔记(18章 处理assets)

    resources资源可以存储声音文件,但当处理多个音乐文件时,效率会很低. assets可以被看作随应用打包的微型文件系统,支持任意层次的文件目录结构.类似游戏这样需要加载大量图片和声音资源的应用通 ...

  3. 安卓权威编程指南-笔记(第21章 XML drawable)

    在Andorid的世界里,凡事要在屏幕上绘制的东西都可以叫drawable,比如抽象图形,Drawable的子类,位图图形等,我们之前用来封装图片的BitmapDrawable就是一种drawable ...

  4. 安卓权威编程指南-笔记(第27章 broadcast intent)

    本章需求:首先,让应用轮询新结果并在有所发现时及时通知用户,即使用户重启设备后还没有打开过应用.其次,保证用户在使用应用时不出现新结果通知. 1. 一般intent和broadcast intent ...

  5. 安卓权威编程指南-笔记(第23章 HTTP与后台任务)

    1. 网络连接基本 //通过指定URL获取原始数据,并返回一个字节流数组. public byte[] getUrlBytes(String urlSpec)throws IOException{ / ...

  6. 安卓权威编程指南-笔记(第22章 深入学习intent和任务)

    本章,我们会使用隐式intent创建一个替换android默认启动器的应用.名为NerdLauncher. NerdLauncher应用能列出设备上的其他应用,点选任意列表项会启动相应应用. 1. 解 ...

  7. 安卓权威编程指南 挑战练习 25章 深度优化 PhotoGallery 应用

    你可能已经注意到了,提交搜索时, RecyclerView 要等好一会才能刷新显示搜索结果.请接受挑战,让搜索过程更流畅一些.用户一提交搜索,就隐藏软键盘,收起 SearchView 视图(回到只显示 ...

  8. 安卓权威编程指南 - 第五章学习笔记(两个Activity)

    学习安卓编程权威指南第五章的时候自己写了个简单的Demo来加深理解两个Activity互相传递数据的问题,然后将自己的学习笔记贴上来,如有错误还请指正. IntentActivityDemo学习笔记 ...

  9. 安卓权威编程指南 挑战练习(第26章 在 Lollipop 设备上使用 JobService)

    26.11 挑战练习:在 Lollipop 设备上使用 JobService 请创建另一个 PollService 实现版本.新的 PollService 应该继承 JobService 并使用 Jo ...

随机推荐

  1. 20190221 beautiful soup 入门

    beautiful soup 入门 Beautiful Soup 是 python 的一个库,最主要的功能是从网页抓取数据. Beautiful Soup 自动将输入文档转换为 Unicode 编码, ...

  2. 对《The future of ReactiveCocoa》的一些思考

    前言 我以为 第一次接触 swift 语言时,看到函数的表示形式如下: func fun(num: Int) -> Int { return num + 1 } let f = fun(1) 和 ...

  3. 解决IntelliJ IDEA Community 社区版 启动Tomcat插件 "Smart Tomcat" NullPointerException 空指针异常

    IntelliJ IDEA Community社区版默认是没有Ultimate版的Tomcat Server,这时候就可以使用插件"Smart Tomcat"; 在"Ru ...

  4. 吴裕雄--天生自然 pythonTensorFlow自然语言处理:PTB 语言模型

    import numpy as np import tensorflow as tf # 1.设置参数. TRAIN_DATA = "F:\TensorFlowGoogle\\201806- ...

  5. 14 微服务电商【黑马乐优商城】:day04-项目搭建(一)

    本项目的笔记和资料的Download,请点击这一句话自行获取. day01-springboot(理论篇) :day01-springboot(实践篇) day02-springcloud(理论篇一) ...

  6. Python语言学习前提:条件语句

    一.条件语句 1.条件语句:通过一条或多条语句的执行结果(True或False)来决定执行额代码块.python程序语言指定任何非0或非空(null)的值为true,0或null为false. 2. ...

  7. Angular(二)

    Angular开发者指南(二)概念概述   template(模板):带有附加标记的模板HTMLdirectives(指令):使用自定义属性和元素扩展HTMLmodel(模型):用户在视图中显示的数据 ...

  8. day11-random模块-随机

    import random # 一.随机小数: print(random.random()) # 0.848972270116501结果是0-1之间的随机小数 print(random.uniform ...

  9. mysql之存储过程(一)

    今天开发一个需求,需要在一个旧表中增加一列并且对已经的表中记录初始化新列的值, 由于是一次性的工作,故写了个存储过程来代替代码程序初始化 创建及执行过程记录如下: MySQL [XXX_YYY]> ...

  10. 分类算法之KNN分类

    1.介绍 KNN是k nearest neighbor 的简称,即k最邻近,就是找k个最近的实例投票决定新实例的类标.KNN是一种基于实例的学习算法,它不同于贝叶斯.决策树等算法,KNN不需要训练,当 ...