近来找了一些关于android线程间通信的资料,整理学习了一下,并制作了一个简单的例子。

 andriod提供了 Handler 和 Looper 来满足线程间的通信。例如一个子线程从网络上下载了一副图片,当它下载完成后会发送消息给主线程,这个消息是通过绑定在主线程的Handler来传递的。

在Android,这里的线程分为有消息循环的线程和没有消息循环的线程,有消息循环的线程一般都会有一个Looper,这个事android的新 概念。我们的主线程(UI线程)就是一个消息循环的线程。针对这种消息循环的机制,我们引入一个新的机制Handle,我们有消息循环,就要往消息循环里 面发送相应的消息,自定义消息一般都会有自己对应的处理,消息的发送和清除,消息的的处理,把这些都封装在Handle里面,注意Handle只是针对那 些有Looper的线程,不管是UI线程还是子线程,只要你有Looper,我就可以往你的消息队列里面添加东西,并做相应的处理。 
但是这里还有一点,就是只要是关于UI相关的东西,就不能放在子线程中,因为子线程是不能操作UI的,只能进行数据、系统等其他非UI的操作。 
  在Android,这里的线程分为有消息循环的线程和没有消息循环的线程,有消息循环的线程一般都会有一个Looper,这个是android的新概念。我们的主线程(UI线程)就是一个消息循环的线程。针对这种消息循环的机制,我们引入一个新的机制Handler,我们有消息循环,就要往消息循环里面发送相应的消息,自定义消息一般都会有自己对应的处理,消息的发送和清除,把这些都封装在Handler里面,注意Handler只是针对那 些有Looper的线程,不管是UI线程还是子线程,只要你有Looper,我就可以往你的消息队列里面添加东西,并做相应的处理。

但是这里还有一点,就是只要是关于UI相关的东西,就不能放在子线程中,因为子线程是不能操作UI的,只能进行数据、系统等其他非UI的操作。

  一个Handler的创建它就会被绑定到这个线程的消息队列中,如果是在主线程创建的,那就不需要写代码来创建消息队列了,默认的消息队列会在主线程被创建。但是如果是在子线程的话,就必须在创建Handler之前先初始化线程的消息队列。如下面的代码:

Java代码  
  1. class ChildThread extends Thread {
  2. public void run() {
  3. /*
  4. * 创建 handler前先初始化Looper.
  5. */
  6. Looper.prepare();
  7. /*
  8. * 在子线程创建handler,所以会绑定到子线程的消息队列中
  9. *
  10. */
  11. mChildHandler = new Handler() {
  12. public void handleMessage(Message msg) {
  13. /*
  14. * Do some expensive operations there.
  15. */
  16. }
  17. };
  18. /*
  19. * 启动该线程的消息队列
  20. */
  21. Looper.loop();
  22. }
  23. }

当Handler收到消息后,就会运行handleMessage(…)的回调函数,可以在里面做一些耗时的操作。

最后完成了操作要结束子线程时,记得调用quit()来结束消息循环队列。

mChildHandler.getLooper().quit();

下面是一个线程间通信的小例子:

Java代码  
  1. /**
  2. *
  3. * @author allin.dev
  4. * http://allin.cnblogs.com
  5. *
  6. */
  7. public class MainThread extends Activity {
  8. private static final String TAG = "MainThread";
  9. private Handler mMainHandler, mChildHandler;
  10. private TextView info;
  11. private Button msgBtn;
  12. @Override
  13. public void onCreate(Bundle savedInstanceState) {
  14. super.onCreate(savedInstanceState);
  15. setContentView(R.layout.main);
  16. info = (TextView) findViewById(R.id.info);
  17. msgBtn = (Button) findViewById(R.id.msgBtn);
  18. mMainHandler = new Handler() {
  19. @Override
  20. public void handleMessage(Message msg) {
  21. Log.i(TAG, "Got an incoming message from the child thread - "
  22. + (String) msg.obj);
  23. // 接收子线程的消息
  24. info.setText((String) msg.obj);
  25. }
  26. };
  27. new ChildThread().start();
  28. msgBtn.setOnClickListener(new OnClickListener() {
  29. @Override
  30. public void onClick(View v) {
  31. if (mChildHandler != null) {
  32. //发送消息给子线程
  33. Message childMsg = mChildHandler.obtainMessage();
  34. childMsg.obj = mMainHandler.getLooper().getThread().getName() + " says Hello";
  35. mChildHandler.sendMessage(childMsg);
  36. Log.i(TAG, "Send a message to the child thread - " + (String)childMsg.obj);
  37. }
  38. }
  39. });
  40. }
  41. public void onDestroy() {
  42.       super.onDestroy();
  43. Log.i(TAG, "Stop looping the child thread's message queue");
  44. mChildHandler.getLooper().quit();
  45. }
  46. class ChildThread extends Thread {
  47. private static final String CHILD_TAG = "ChildThread";
  48. public void run() {
  49. this.setName("ChildThread");
  50. //初始化消息循环队列,需要在Handler创建之前
  51. Looper.prepare();
  52. mChildHandler = new Handler() {
  53. @Override
  54. public void handleMessage(Message msg) {
  55. Log.i(CHILD_TAG, "Got an incoming message from the main thread - " + (String)msg.obj);
  56. try {
  57. //在子线程中可以做一些耗时的工作
  58. sleep(100);
  59. Message toMain = mMainHandler.obtainMessage();
  60. toMain.obj = "This is " + this.getLooper().getThread().getName() +
  61. ".  Did you send me \"" + (String)msg.obj + "\"?";
  62. mMainHandler.sendMessage(toMain);
  63. Log.i(CHILD_TAG, "Send a message to the main thread - " + (String)toMain.obj);
  64. } catch (InterruptedException e) {
  65. // TODO Auto-generated catch block
  66. e.printStackTrace();
  67. }
  68. }
  69. };
  70. Log.i(CHILD_TAG, "Child handler is bound to - "+ mChildHandler.getLooper().getThread().getName());
  71. //启动子线程消息循环队列
  72. Looper.loop();
  73. }
  74. }
  75. }

转载于:https://www.cnblogs.com/xiaochao1234/p/3922263.html

Thread+Handler 线程 消息循环(转载)相关推荐

  1. Android应用程序线程消息循环模型分析

    出自:http://blog.csdn.net/luoshengyang/article/details/6905587 我们知道,Android应用程序是通过消息来驱动的,即在应用程序的主线程(UI ...

  2. Android应用程序线程消息循环模型分析(5)

      从AsyncTask的实现可以看出,当我们第一次创建一个AsyncTask对象时,首先会执行下面静态初始化代码创建一个线程池sExecutor: private static final Bloc ...

  3. 图解 Android Handler 线程消息机制

    从现实生活中理解线程消息机制 android 有一种叫消息队列的说法,这里我们可以这样理解:假如一个隧道就是一个消息队列,那么里面的每一部汽车就是一个一个消息,这里我们先忽略掉超车等种种因素,只那么先 ...

  4. Android应用程序线程消息循环模型分析(4)

    接下来我们再看看应用程序的配置文件AndroidManifest.xml: <?xml version="1.0" encoding="utf-8"?&g ...

  5. android 消息循环机制--looper handler

    Looper类说明   Looper 类用来为一个线程跑一个消息循环. 线程在默认情况下是没有消息循环与之关联的,Thread类在run()方法中的内容执行完之后就退出了,即线程做完自己的工作之后就结 ...

  6. 模态对话框的消息循环原理及分析笔记

    简述: APP消息循环和模态对话框中局部消息循环的关系 根据上图可以看出,在APP的消息循环再派发ONOK消息后,调用ModalDlg的响应函数,pWnd->OnOk();在该消息中, 会 进入 ...

  7. 模态对话框和非模态对话框的消息循环分析

    1.非模态对话框和父窗口共享当前线程的消息循环 2.模态对话框新建一个新的消息循环,并由当前消息循环派发消息,而父窗口.模态对话框屏蔽了用户对它父窗口的操作,但是不是在消息循环里面屏蔽,所以给父窗口发 ...

  8. android 结束if循环_Android Handler 消息循环机制

    前言 一问起Android应用程序的入口,很多人会说是Activity中的onCreate方法,也有人说是ActivityThread中的静态main方法.因为Java虚拟机在运行的时候会自动加载指定 ...

  9. Android异步机制一:使用Thread+Handler实现非UI线程更新UI界面

    概述:每个Android应用程序都运行在一个dalvik虚拟机进程中,进程开始的时候会启动一个主线程(MainThread),主线程负责处理和ui相关的事件,因此主线程通常又叫UI线程.而由于Andr ...

  10. Android的消息循环机制:Handler

    前言 Android的消息机制主要是指Handler的运行机制,对于大家来说Handler已经是轻车熟路了,可是真的掌握了Handler?本文主要通过几个问题围绕着Handler展开深入并拓展的了解. ...

最新文章

  1. python下载电脑版本不对_初学Python,因为某些原因电脑只能装3.1版本,现遇到这个小问题求解答...
  2. 树——二叉搜索树的实现
  3. 中英文最大AI模型世界纪录产生,大模型竞赛新阶段来了
  4. AFNetworking 3.0 源码解读(一)之 AFNetworkReachabilityManager
  5. 本科生学习fpga,dsp,嵌入式操作系统哪个好
  6. python算法与数据结构-顺序表(39)
  7. [css] 举例说明如何从html元素继承box-sizing?
  8. 【指针】C++中指针的使用艺术[转]
  9. git忽略文件或者文件夹
  10. python适合自学编程吗-对没有编程基础的人来说,直接学Python入门IT合适吗?
  11. Tomcat 配置虚拟目录
  12. 【视频分享】尚硅谷Oracle视频教程
  13. 在职工程师如何在乾颐堂通过HCIE认证考试一瞥(王IE战报)
  14. 手机计算机如何用科学计算法,手机计算器开根号怎么按(万能科学计算器在线使用方法)...
  15. Linux性能分析命令
  16. linux下 Apache 配置虚拟主机三种方式
  17. Hexo - Next - Mist 风格主题的美化(二)
  18. win10系统怎样彻底关闭更新
  19. OpenStack组件部署之Placement
  20. 光量子计算机的功能,什么是光计算机和量子计算机?

热门文章

  1. django框架搭建网页后台,运行后网页打不开的解决方法--windows系统下
  2. 无向图是欧拉图的充要条件_500页开放书搞定概率图建模,图灵奖得主Judea Pearl推荐...
  3. cocos2dx 字体外发光_Cocos2d-x 3.x开发塔防游戏《王国保卫战》02:地图(一)
  4. 力扣-811 子域名访问计数
  5. react 使用cookie react-cookies
  6. Java — Arrays.toString(string[]) + Enumeration.hasMoreElements()【Arrays类、Enumeration接口】
  7. Vue.js05:vue内联样式
  8. PostgreSQL 在Ubuntu下如何修改postgres默认密码
  9. Nginx 高级配置
  10. 寻访x86处理器“实模式”和“保护模式”的前世今生