转载请标明出处:一片枫叶的专栏

上一篇文章中我们解说了AsyncTast的基本使用以及实现原理,我们知道AsyncTask内部是通过线程池和Handler实现的。通过对线程池和handler的封装实现了对异步任务操作。很多其它关于AsyncTask相关的内容,可參考我的android源代码解析之(三)–>异步任务AsyncTask

本文我们将解说HandlerThread相关的概念。

HandlerThread是什么东西呢?了解一个类最好的方法就是查看类的定义,所以我们就看一下HandlerThread是怎样定义的吧。

查看类的定义时有这样一段话:

Handy class for starting a new thread that has a looper.
The looper can then be used to create handler classes.
Note that start() must still be called.

意思就是说:这个类的作用是创建一个包括looper的线程。

那么我们在什么时候须要用到它呢?增加在应用程序其中为了实现同一时候完毕多个任务,所以我们会在应用程序其中创建多个线程。为了让多个线程之间能够方便的通信,我们会使用Handler实现线程间的通信。

这个时候我们手动实现的多线程+Handler的简化版就是我们HandlerThrea所要做的事了。

以下我们首先看一下HandlerThread的基本使用方法:

/**
* 測试HandlerThread的基本使用
*/
HandlerThread mHandlerThread = new HandlerThread("myHandlerThreand");
mHandlerThread.start(); // 创建的Handler将会在mHandlerThread线程中运行
final Handler mHandler = new Handler(mHandlerThread.getLooper()) {
@Override
public void handleMessage(Message msg) {
Log.i("tag", "接收到消息:" + msg.obj.toString());
}
}; title = (TextView) findViewById(R.id.title);
title.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) { Message msg = new Message();
msg.obj = "11111";
mHandler.sendMessage(msg); msg = new Message();
msg.obj = "2222";
mHandler.sendMessage(msg);
}
});

我们首先定义了一个HandlerThread对象,是直接通过new的方式产生的,查看其构造方法:

public HandlerThread(String name) {
super(name);
mPriority = Process.THREAD_PRIORITY_DEFAULT;
}

能够知道HandlerThread继承于Thread。所以说HandlerThread本质上是一个线程。其构造方法主要是做一些初始化的操作。

然后我们调用了mHandlerThread.start()方法。由上我们知道了HandlerThread类事实上就是一个Thread。一个线程,所以其start方法内部调用的肯定是Thread的run方法,我们查看一下其run方法的详细实现:

@Override
public void run() {
mTid = Process.myTid();
Looper.prepare();
synchronized (this) {
mLooper = Looper.myLooper();
notifyAll();
}
Process.setThreadPriority(mPriority);
onLooperPrepared();
Looper.loop();
mTid = -1;
}

我们发现其内部调用了Looper.prepate()方法和Loop.loop()方法。熟悉android异步消息机制的童鞋应当知道。在android体系中一个线程事实上是相应着一个Looper对象、一个MessageQueue对象。以及N个Handler对象,详细可參考:android源代码解析之(二)–>异步消息机制

所以通过run方法,我们能够知道在我们创建的HandlerThread线程中我们创建了该线程的Looper与MessageQueue。

这里须要注意的是其在调用Looper.loop()方法之前调用了一个空的实现方法:onLooperPrepared(),我们能够实现自己的onLooperPrepared()方法,做一些Looper的初始化操作。

run方法里面当mLooper创建完毕后有个notifyAll(),getLooper()中有个wait(),这是为什么呢?由于的mLooper在一个线程中运行。而我们的handler是在UI线程初始化的,也就是说,我们必须等到mLooper创建完毕,才干正确的返回getLooper();wait(),notify()就是为了解决这两个线程的同步问题

然后我们调用了:

// 创建的Handler将会在mHandlerThread线程中运行
final Handler mHandler = new Handler(mHandlerThread.getLooper()) {
@Override
public void handleMessage(Message msg) {
Log.i("tag", "接收到消息:" + msg.obj.toString());
}
};

该Handler的构造方法中传入了HandlerThread的Looper对象,所以Handler对象就相当于含有了HandlerThread线程中Looper对象的引用。

然后我们调用handler的sendMessage方法发送消息,在Handler的handleMessge方法中就能够接收到消息了。

最后须要注意的是在我们不须要这个looper线程的时候须要手动停止掉;

protected void onDestroy() {
super.onDestroy();
mHandlerThread.quit();
}

好了,以上就是HandlerThread相关的知识了。相对来说HandlerThread还是比較简单的。其本质上就是一个Thread线程,仅仅只是里面包括了Looper和MessageQueue,最后我们在这里总结一下。

总结:

  • HandlerThread本质上是一个Thread对象。仅仅只是其内部帮我们创建了该线程的Looper和MessageQueue;

  • 通过HandlerThread我们不但能够实现UI线程与子线程的通信相同也能够实现子线程与子线程之间的通信。

  • HandlerThread在不须要使用的时候须要手动的回收掉;

另外对android源代码解析方法感兴趣的可參考我的:

android源代码解析之(一)–>android项目构建过程

android源代码解析之(二)–>异步消息机制

android源代码解析之(三)–>异步任务AsyncTask


本文以同步至github中:https://github.com/yipianfengye/androidSource,欢迎star和follow


Android源代码解析之(四)-->HandlerThread的更多相关文章

  1. Android源代码解析之(六)-->Log日志

    转载请标明出处:一片枫叶的专栏 首先说点题外话,对于想学android framework源代码的同学,事实上能够在github中fork一份,详细地址:platform_frameworks_bas ...

  2. Android源代码解析之(七)-->LruCache缓存类

    转载请标明出处:一片枫叶的专栏 android开发过程中常常会用到缓存.如今主流的app中图片等资源的缓存策略通常是分两级.一个是内存级别的缓存,一个是磁盘级别的缓存. 作为android系统的维护者 ...

  3. Android源代码解析之(十三)-->apk安装流程

    转载请标明出处:一片枫叶的专栏 上一篇文章中给大家分析了一下android系统启动之后调用PackageManagerService服务并解析系统特定文件夹.解析apk文件并安装的过程,这个安装过程实 ...

  4. Android源代码解析之(三)-->异步任务AsyncTask

    转载请标明出处:一片枫叶的专栏 上一篇文章中我们解说了android中的异步消息机制. 主要解说了Handler对象的使用方式.消息的发送流程等.android的异步消息机制是android中多任务处 ...

  5. Android 源代码解析 之 setContentView

    转载请标明出处:http://blog.csdn.net/lmj623565791/article/details/41894125,本文出自:[张鸿洋的博客] 大家在平时的开发中.对于setCont ...

  6. Android中HandlerThread的使用及源代码解析

    关于Hanlder的基本使用能够參见博文<Android中Handler的使用>,假设想了解Handler.Looper.Thread等的相互关系以及内部实现原理能够參见博文<深入源 ...

  7. 《Android源代码设计模式解析与实战》读书笔记(十四)

    第十四章.迭代器模式 迭代器模式,又叫做游标模式.是行为型设计模式之中的一个.我们知道对容器对象的訪问必定会涉及遍历算法.我们能够将遍历的方法封装在容器中,或者不提供遍历方法,让使用容器的人自己去实现 ...

  8. Android EventBus源代码解析 带你深入理解EventBus

    转载请标明出处:http://blog.csdn.net/lmj623565791/article/details/40920453,本文出自:[张鸿洋的博客] 上一篇带大家初步了解了EventBus ...

  9. 源代码解析Android中View的layout布局过程

    Android中的Veiw从内存中到呈如今UI界面上须要依次经历三个阶段:量算 -> 布局 -> 画图,关于View的量算.布局.画图的整体机制可參见博文 < Android中Vie ...

随机推荐

  1. 认识createDocumentFragment

    今天在看vue源码解析时候发现一个api没有见过,一查是原生的,赶紧补漏. DocumentFragments 是DOM节点.它们不是主DOM树的一部分.通常的用例是创建文档片段,将元素附加到文档片段 ...

  2. Codeforces #430 Div2 C

    #430 Div2 C 题意 给出一棵带点权的树,每一个节点的答案为从当前节点到根节点路径上所有节点权值的最大公因子(在求最大共因子的时候可以选择把这条路径上的任意一点的权值置为0).对于每一个节点单 ...

  3. POJ1330 Nearest Common Ancestors(最近公共祖先)(tarjin)

    A - Nearest Common Ancestors Time Limit:1000MS     Memory Limit:10000KB     64bit IO Format:%lld &am ...

  4. 【Android】 HttpClient 发送REST请求

    直接po代码吧,第一个是一个枚举类型的类,是四种rest http请求,get/post/put/delete: public enum HttpRequestMethod { HttpGet { @ ...

  5. Android的日志工具Log

    Android中的日志工具类是Log(android.util.Log),这个类提供了以下几个方法来供我们打印日志. ♦ Log.v():这个方法用于打印那些最为琐碎的,意义最小的日志信息.对应级别v ...

  6. struts2 action 字段问题

    struts2最多只能解释两级字段,比如user.username,像user.info.age在类中属性类的三段字符不能识别,只能先用user,info 然后在user.setInfo(info);

  7. [CF930E]/[CF944G]Coins Exhibition

    [CF930E]/[CF944G]Coins Exhibition 题目地址: CF930E/CF944G 博客地址: [CF930E]/[CF944G]Coins Exhibition - skyl ...

  8. python基础-匿名函数、内置函数、正则表达式、模块

    1. 匿名函数 1.1 有名函数 有名函数:定义了一个函数名,函数名指向内存地址:通过函数名进行访问.函数名加括号就可以运行有名函数,例如:func() def func(x, y, z = 1): ...

  9. ios-真机调试出错信息

    更新证书错误Code Sign error: Provisioning profile ‘XXXX'can't be found                   在Xcode中当你在更新了你得证书 ...

  10. 全局流水ID号生成的几种方法

    这个问题源自于,我想找一个分布式下的ID生成器.  这个最简单的方案是,数据库自增ID.为啥不用咧?有这么几点原因,一是,会依赖于数据库的具体实现,比如,mysql有自增,oracle没有,得用序列, ...