刚开始学习Android的时候,知道异步线程无法更新UI,于是找了个能把更新的动作抛给UI线程的东西,这个东西就是Handler。

一开始就只会在主线程也就是UI线程new一个Handler,之后在各个子线程里面使用,并没想过一些原理的东西,其实需要学习的知识还有很多。

一、线程之间的同步

A. 子线程向主线程发送消息,我们一开始学习的都是这种比较简单方式。

1、主线程中new Handler,并实现handleMessage方法

2、子线程中获得主线程Handler的实例

3、子线程向主线程发消息sendMessage

B. 主线程向子线程发送消息:

#点击按钮向mThread线程发送消息
mStartBtn.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
......
mHandler.sendMessage(......);
}
});
mThread.start();
...... Thread mThread = new Thread(new Runnable() {
@Override
public void run() {
Looper.prepare();
#实例化Handler,实现handleMessage
mHandler = new Handler(){
@Override
public void handleMessage(Message msg) {
......
}
};
Looper.loop();
}
});

通过这个我们比较直观的看见相同和不同的地方:

相同的地方:A线程向B线程发送消息,A需要拿到B实例化的Handler对象

不同的地方:子线程多了下面两个东西:

Looper.prepare();
.......
Looper.loop();

其实说不同的地方,只是A类型我们实现的时候不需要写Looper,理论上主线程一样需要有Looper,那么很容易想到,Android已经给我们写好了,后面有具体讲。

二、Looper,Handler,Message

Looper:

    private Looper(boolean quitAllowed) {
mQueue = new MessageQueue(quitAllowed);
mThread = Thread.currentThread();
} public static void prepare() {
prepare(true);
} private static void prepare(boolean quitAllowed) {
if (sThreadLocal.get() != null) {
throw new RuntimeException("Only one Looper may be created per thread");
}
sThreadLocal.set(new Looper(quitAllowed));
}

Looper在创建的时候会初始化一个MessageQueue,这个是用来存储Message的管道

    public static void loop() {
final Looper me = myLooper();
if (me == null) {
throw new RuntimeException("No Looper; Looper.prepare() wasn't called on this thread.");
}
final MessageQueue queue = me.mQueue;
...... for (;;) {
Message msg = queue.next(); // might block
......
final long end;
try {
msg.target.dispatchMessage(msg);
end = (slowDispatchThresholdMs == 0) ? 0 : SystemClock.uptimeMillis();
} finally {
if (traceTag != 0) {
Trace.traceEnd(traceTag);
}
}
......
}
}

在loop的时候会循环的从MessageQueue取Message,然后分发Message。msg.target其实就是Handler,于是Handler就可以获得数据并进行处理。

    public void dispatchMessage(Message msg) {
if (msg.callback != null) {
handleCallback(msg);
} else {
if (mCallback != null) {
if (mCallback.handleMessage(msg)) {
return;
}
}
handleMessage(msg);
}
} private static void handleCallback(Message message) {
message.callback.run();
}

上面的还是比较有意思的,这里简单总结下:

  1. msg.callback不为空,执行msg.callback.run
  2. mCallback不为null,执行mCallback.handleMessage(msg),可能会执行第3中情况
  3. 最后可能会执行Handler本身的handleMessage(msg)方法。

一般的是3这种情况,1发生在使用Handler在异步线程中直接更新UI的情况。

    Thread mThread = new Thread(new Runnable() {
@Override
public void run() {
mHandler.post(new Runnable() {
@Override
public void run() {
textView.setText("xxxx");
}
}); }
});

其实mHandler.post也是发送了一个消息,然后把Runnable传给Message并处理。

总结下:

Looper其实就是创建一个循环获取消息并发送Handler的类。

Message其实就是一个数据存储的类,用于传输数据。

Handler就是进行发送和处理消息的类。

这三个配合起来一起用,才能构建了handler线程之间传递数据的机制。

问题一:上面所说的需要Looper,Handler,Message配合使用才能完成这个工作,那么我写子线程向主线程时怎么没看到主线程的Looper呢?

这个我们可以去看下Activity的源码,其实在ActivityThread里面已经写好了这个东西,所有我们不需要写了。

public static void main(String[] args) {
...... Process.setArgV0("<pre-initialized>"); Looper.prepareMainLooper(); ActivityThread thread = new ActivityThread();
thread.attach(false); ......
Looper.loop(); throw new RuntimeException("Main thread loop unexpectedly exited");
}

注意:写在Looper.loop();后面的代码是无法执行。

Handler学习的更多相关文章

  1. 【转】Handler学习笔记(二)

    一.一个问题 有这样一个问题值得我们思考,若把一些类似于下载的功能(既耗时且不一定有结果)写在Activity(主线程)里,会导致Activity阻塞,长时间无响应,直至页面假死(如果5秒钟还没有完成 ...

  2. 【转】Handler学习笔记(一)

    一.Handler的定义: Handler主要接收子线程发送的数据, 并用此数据配合主线程更新UI,用来跟UI主线程交互用.比如可以用handler发送一个message,然后在handler的线程中 ...

  3. Handler学习小结

    在android消息机制中Handler扮演着举足轻重的作用,(AsnyTask其实也是对Handler+Thread做了一层封装),ui线程超过5s就会报出ANR,一般耗时操作操作需要放在子线程中处 ...

  4. 2018-03-17 handler学习使用

    1.handler具体使用https://www.cnblogs.com/JohnTsai/p/5259869.html 2.handlerThread用法https://www.jianshu.co ...

  5. [转]Handler学习笔记(二)

    一.一个问题 有这样一个问题值得我们思考,若把一些类似于下载的功能(既耗时且不一定有结果)写在Activity(主线程)里,会导致Activity阻塞,长时间无响应,直至页面假死(如果5秒钟还没有完成 ...

  6. [转]Handler学习笔记(一)

    一.Handler的定义: Handler主要接收子线程发送的数据, 并用此数据配合主线程更新UI,用来跟UI主线程交互用.比如可以用handler发送一个message,然后在handler的线程中 ...

  7. Android Handler学习笔记

    已经习惯了挖坑不填,继续任性一下,周一到周五继续挖坑,每周六周日负责填坑. 1.从Android UI线程谈起 出于性能考虑,Android 中的UI操作并不是线程安全的,所以Android中规定只能 ...

  8. Android学习笔记-事件处理之Handler消息传递机制

    内容摘要:Android Handler消息传递机制的学习总结.问题记录 Handler消息传递机制的目的: 1.实现线程间通信(如:Android平台只允许主线程(UI线程)修改Activity里的 ...

  9. Android精通之Handler讲解

    版权声明:未经博主允许不得转载 一:简介 [达叔有道]软件技术人员,时代作者,从 Android 到全栈之路,我相信你也可以!阅读他的文章,会上瘾!You and me, we are family ...

随机推荐

  1. OSGi-入门篇之生命周期层(03)

    前言 生命周期层在OSGi框架中属于模块层上面的一层,它的运作是建立在模块层的功能之上的.生命周期层一个主要的功能就是让你能够从外部管理应用或者建立能够自我管理的应用(或者两者的结合),并且给了应用本 ...

  2. 软件工程个人第二小项目——wc

    github源码和工程文件地址:https://github.com/HuChengLing/wc 基本要求:要实现wc的基本功能即文件中字符数.单词数.行数的统计. 主要功能:文件中字符数.单词数. ...

  3. 怎样使用自定义标签简化 js、css 引入?

    国庆将至,工作兴致全无,来总结点项目里平时不起眼干货. 前端引入 js .css 一般是这样: <script type="text/javascript" src=&quo ...

  4. Ubuntu16笔记本双显卡安装NVIDIA驱动

    blockquote { direction: ltr; color: rgb(0, 0, 0) } blockquote.western { font-family: "Liberatio ...

  5. 框架应用 : Spring开发详述

    Spring framework简介 spring framework这个框架是spring项目中的核心项目,所有框架都依赖于这个框架. 它是一个一站式的开源框架,基础技术是IoC. 按官方文档主要分 ...

  6. VB.net shell、IO.File.Open、Process.Start、Shellexecute API 运用经验总结

    打开文件还有很多方法,但我了解到运用较多的是上面几种- -,为了防止以后忘记,先把了解到的写下来. 1.Shell 这个看了很多网页,最靠谱的运用方法: Shell("cmd.exe /c ...

  7. Oracle添加含有脏数据的约束

    需求: 一个表的唯一约束被禁用期间,有脏数据进来,当启用约束时失败. 环境: -bash-4.1$ uname -a Linux dbtest1 2.6.32-279.el6.x86_64 #1 SM ...

  8. C# readonly

    当某个字段是引用类型,并且该字段被标记为readonly时,不可改变的是引用,而非字段引用的对象,例如:

  9. ZOJ2105 终于找到错误

    ZOJ2105:点击打开链接 错误代码 #include<stdio.h> #include<stdlib.h> int q[110]; int main() { int a, ...

  10. 提纲挈领webrtc之vad检测

    顾名思义,VAD(Voice Activity Detection)算法的作用是检测是否是人的语音,它的使用 范围极广,降噪,语音识别等领域都需要有vad检测.vad检测有很多方法,这里我们之介绍一 ...