近来找了一些关于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之前先初始化线程的消息队列。如下面的代码:

  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();

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

  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. }

Thread+Handler 线程 消息循环(转载)的更多相关文章

  1. Android----Thread+Handler 线程 消息循环(转载)

    近来找了一些关于android线程间通信的资料,整理学习了一下,并制作了一个简单的例子. andriod提供了 Handler 和 Looper 来满足线程间的通信.例如一个子线程从网络上下载了一副图 ...

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

    文章转载至CSDN社区罗升阳的安卓之旅,原文地址:http://blog.csdn.net/luoshengyang/article/details/6905587 我们知道,Android应用程序是 ...

  3. Android之MessageQueue、Looper、Handler与消息循环

    在android的activity中有各种各样的事件,而这些事件最终是转换为消息来处理的.android中的消息系统涉及到: *  消息发送 *  消息队列 *  消息循环 *  消息分发 *  消息 ...

  4. Android的消息循环机制 Looper Handler类分析

    Android的消息循环机制 Looper Handler类分析 Looper类说明   Looper 类用来为一个线程跑一个消息循环. 线程在默认情况下是没有消息循环与之关联的,Thread类在ru ...

  5. Android线程消息通信(一)

    Android在Java标准线程模型的基础上,提供了消息驱动机制,用于多线程之间的通信.基于消息驱动机制的线程通信模型陈伟线程消息通信.在标准线程模型中,线程执行完毕后便退出,而Android扩展了线 ...

  6. Chromium on Android: Android在系统Chromium为了实现主消息循环分析

    总结:刚开始接触一个Chromium on Android时间.很好奇Chromium主消息循环是如何整合Android应用. 为Android计划,一旦启动,主线程将具有Java消息层循环处理系统事 ...

  7. System、应用程序进程的Binder线程池和Handler消息循环

    首先看一张Android系统启动流程图:

  8. [转]Handler MessageQueue Looper消息循环原理分析

    Handler MessageQueue Looper消息循环原理分析   Handler概述 Handler在Android开发中非常重要,最常见的使用场景就是在子线程需要更新UI,用Handler ...

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

    Android应用的开发过程中需要把繁重的任务(IO,网络连接等)放到其他线程中异步执行,达到不阻塞UI的效果. 下面将由浅入深介绍Android进行异步处理的实现方法和系统底层的实现原理. 本文介绍 ...

随机推荐

  1. [Everyday Mathematics]20150122

    设 $f:[0,1]\to [0,1]$. (1). 若 $f$ 连续, 试证: $\exists\ \xi\in [0,1],\st f(\xi)=\xi$. (2). 若 $f$ 单调递增, 试证 ...

  2. ylbtech-权限管理-数据库设计-功能权限管理技术

    ylbtech-DatabaseDesgin:ylbtech-权限管理-数据库设计-功能权限管理技术 DatabaseName:ylb_permission(权限管理-功能权限管理技术)实现 Type ...

  3. iOS数据存储之属性列表理解

    iOS数据存储之属性列表理解 数据存储简介 数据存储,即数据持久化,是指以何种方式保存应用程序的数据. 我的理解是,开发了一款应用之后,应用在内存中运行时会产生很多数据,这些数据在程序运行时和程序一起 ...

  4. ndk文件操作问题及小结

    最近在做文件传输,发现在android下用f系列的C库函数去读取文件文件大小会受到2G大小的约束,查阅了很久,最后只能去看google的libc源码,发现了以下几个问题: 1.bionic的libc是 ...

  5. convert source code files to pdf format in python

    import os import sys def find_file(root_dir, type): dirs_pool = [root_dir] dest_pool = [] def scan_d ...

  6. spoj 839 Optimal Marks(二进制位,最小割)

    [题目链接] http://acm.hust.edu.cn/vjudge/problem/viewProblem.action?id=17875 [题意] 给定一个图,图的权定义为边的两端点相抑或值的 ...

  7. FLEX实现两侧边栏固定中间自适应布局

    <style type="text/css"> #outer{ display: flex; width: 100%; flex-flow: row nowrap; } ...

  8. vb调用exe文件

    vb调用exe文件 函数:Call Shell(PathName,WindowStyle) 或 a = Shell(PathName,WindowStyle) ,不需要声明. 注解:PathName ...

  9. homework-03

    1.分工准备 这次的工作是结对编程,在第二次作业中我是使用python完成的作业,而小明是使用C完成的作业.因为打算使用动态链接库的方式将第二次的代码嵌入到本次的作业中,而python生成动态链接库不 ...

  10. sizeof 字符数组

    比较 #include <stdio.h> #include <string.h> int main(int argc, const char *argv[]) { char ...