【Android 开发】: Android 消息处理机制之一: Handler 与 Message

最近几讲内容,我们学习了Android中关于多线程的一些知识,上一讲我们讲解了异步任务 AsyncTask 的操作,Android中还提供了其他的线程操作,如Handler Message MessageQueue Looper 等模块,这些就是Android中的消息处理机制。这部分内容是Android学习过程中的重点和难点。
    现在我们就来学习一下Android的消息处理,以及剖析一下相关类如Handler和Message类的源代码,同时使用他们来更新UI主线程的操作。因为Android的消息处理机制内容繁多,我们分为几部分来学习,大家可以关注这几讲内容,这一讲我们重点来学习一下Handler和Message.

一. Handler类介绍

1). 查看Android官网API Handler类

java.lang.Object
    android.os.Handler
Known Direct Subclasses
AsyncQueryHandler, AsyncQueryHandler.WorkerHandler, HttpAuthHandler, SslErrorHandler

一个Handler会允许你发送和处理Message或者Runnable对象关联到一个线程的消息队列MessageQueue中,每一个Handler的实例都会关联一个单一的线程和那个线程的消息队列中。当你创建一个一个新的Handler,它会绑定到你创建的线程和这个线程消息队列中。并且指向好它,它会让消息传递到关联好它的消息队列中,当它从消息队列出队的时候执行它。这里他们的如何关联的不是很懂!

对于Handler来说有两种主要的方式: 1. 计划好消息和Runnable将来的某一个时间点来执行它 2. 从一个不同的线程中执行Handler的入队操作。分发消息由下面的几个方法完成:

1) post(Runnable),
   2) postAtTime(Runnable, long), 
   3) postDelayed(Runnable, long), 
   4) sendEmptyMessage(int), 
   5) sendMessage(Message), 
   6) sendMessageAtTime(Message, long), 
   7) sendMessageDelayed(Message, long)

post方式的方法可以将一个Runable对象排列到消息队列中。sendMessage方式的方法可以通过 Handler的handleMessage(Message) 方法携带有bundle类型的数据的Message对象到队列中(需要你实现Handler的子类)。你可以通过上诉两种方式来出来Handler,你可以允许你的消息在消息队列中准备好就马上被处理,也可以处理之前指定一些延时让你实现超时或者基于时间的行为。
   当你的应用程序的进程被创建的时候,它的主线程专门用来处理正常运行的主线程的消息队列,(也就是说UI主线程有自己的消息队列,所以我们没必要在UI主线程中处理自己的消息)它关心的是管理顶层的应用对象(activities, broadcast receivers, etc)和他们创建的窗口。你可以创建你自己的线程,然后通过Handler与主线程沟通。就像上述说的通过post和sendMessage的方式,Runnable和Message会被计划的执行在Handler的消息队列中适时的进行处理。

二. Message类介绍

1). 查看Android官网API Message类

Java.lang.Object
       android.os.Message

定义一个message包含描述信息和任意的数据对象发送给Handler。这个对象包含两个额外的int类型的属性和一个Object类型的属性,它可以让你不需要去做一些强制类型的转换的操作。如下图所示:

1) arg1 和 arg2 都是Message自带的用来传递一些轻量级存储int类型的数据,比如进度条的数据等。通过这个数据是通过Bundle的方式来转载的,读者可以自己查阅源代码研究。

2) obj 是Message自带的Object类型对象,用来传递一些对象。兼容性最高避免对齐进行类型转换等。

3) replyTo 是作为线程通信的时候使用.

4) what 用户自定义的消息码让接受者识别消息种类,int类型。

【注意】: 获得Message的构造方法最好的方式是调用Message.obtain() 和 Handler.obtainMessage()方法。以便能够更好被回收池所回收[这里读者可以研究一下obtain()的源代码即可明白]。而不是直接用 new Message的方式来获得Message对象。

三. 程序Demo

1. 实现通过 Thread + Handler + Message 的方式下载网络数据。程序结构如下图所示

2. 在Manifest.xml中添加网络权限,这里不再贴出,读者可以参考上面一讲内容

3. 布局文件中 activity_main.xml 中定义Button和ImageView控件,这里不再贴出,读者可以自己下载源代码查看

4. MainActivity.java 程序主代码

[java] view plain copy

  1. <span style="font-family:Courier New;">...</span>
  2. /**
  3. * 通过 Handler + Message 的方式下载网络数据
  4. * 通过子线程run()方法中下载数据,使用Message携带数据,然后用Handler发送消息并且处理消息来更新UI.
  5. *
  6. * @author AHuier
  7. */
  8. public class MainActivity extends Activity {
  9. private Button btn;
  10. private ImageView imageView;
  11. private String imgPath = "http://f.hiphotos.baidu.com/image/w%3D2048/sign=05793c21bba1cd1105b675208d2ac9fc/43a7d933c895d14350ee3c3272f082025aaf0703.jpg";
  12. private static final int DOWNLOAD_IMG = 1;
  13. private ProgressDialog dialog = null;
  14. private Handler handler = new Handler() {
  15. // 处理子线程给我们发送的消息。
  16. @Override
  17. public void handleMessage(android.os.Message msg) {
  18. byte[] data = (byte[])msg.obj;
  19. Bitmap bitmap = BitmapFactory.decodeByteArray(data, 0, data.length);
  20. imageView.setImageBitmap(bitmap);
  21. if(msg.what == DOWNLOAD_IMG){
  22. dialog.dismiss();
  23. }
  24. };
  25. };
  26. @Override
  27. protected void onCreate(Bundle savedInstanceState) {
  28. super.onCreate(savedInstanceState);
  29. setContentView(R.layout.activity_main);
  30. initComponent();
  31. btn.setOnClickListener(new View.OnClickListener() {
  32. @Override
  33. public void onClick(View v) {
  34. // TODO Auto-generated method stub
  35. new Thread(new MyThread()).start();
  36. dialog.show();
  37. }
  38. });
  39. }
  40. @Override
  41. public boolean onCreateOptionsMenu(Menu menu) {
  42. // Inflate the menu; this adds items to the action bar if it is present.
  43. getMenuInflater().inflate(R.menu.main, menu);
  44. return true;
  45. }
  46. private void initComponent() {
  47. btn = (Button) this.findViewById(R.id.button1);
  48. imageView = (ImageView) this.findViewById(R.id.imageView1);
  49. dialog = new ProgressDialog(this);
  50. dialog.setTitle("提示");
  51. dialog.setMessage("正在下载,请稍后...");
  52. dialog.setCancelable(false);
  53. }
  54. // 使用Handler Message MessageQueue Looper等方式去访问网络资源的时候,我们必须要开启一个子线程
  55. public class MyThread implements Runnable{
  56. // 在run方法中完成网络耗时的操作
  57. @Override
  58. public void run() {
  59. HttpClient httpClient = new DefaultHttpClient();
  60. HttpGet httpGet = new HttpGet(imgPath);
  61. HttpResponse httpResponse = null;
  62. try {
  63. httpResponse = httpClient.execute(httpGet);
  64. if(200 == httpResponse.getStatusLine().getStatusCode()){
  65. byte[] data = EntityUtils.toByteArray(httpResponse.getEntity());
  66. // 这里的数据data我们必须发送给UI的主线程,所以我们通过Message的方式来做桥梁。
  67. Message message = Message.obtain();
  68. message.obj = data;
  69. message.what = DOWNLOAD_IMG;
  70. handler.sendMessage(message);
  71. }
  72. } catch (Exception e) {
  73. // TODO: handle exception
  74. }
  75. }
  76. }
  77. }

5. 程序执行结果

四、程序Demo总结

1. 上述Demo中的Handler写法我们只是按图索骥的方式实现比较标准的写法,如果考虑android内存机制的情况下,private Handler的方式定义成为静态的会更好。在Android源码中Handler一般是定义成 protect 权限的。

  2.Handler主要是用来负责发送消息和处理消息的。

3.基于处理,其实这里面蕴含着一个消息队列的概念,这里为什么我们获得消息需要用到Obtain()的方式,而不是通过new的方式构建一个消息,这个问题下一讲我们会通过剖析Message中Obtain()的源代码来讨论。

更多关于Android消息处理机制请点击以下相关链接:

1. Android 消息处理机制之二: Message中obtain()源代码剖析

2. Android 消息处理机制之三: Handler中sendMessage()源代码剖析

源码下载地址:HandlerMessageTest

原文转自:http://blog.csdn.net/ahuier/article/details/17012005

原作者为 AHuier.  请尊重原作者版权

最近几讲内容,我们学习了Android中关于多线程的一些知识,上一讲我们讲解了异步任务 AsyncTask 的操作,Android中还提供了其他的线程操作,如Handler Message MessageQueue Looper 等模块,这些就是Android中的消息处理机制。这部分内容是Android学习过程中的重点和难点。
    现在我们就来学习一下Android的消息处理,以及剖析一下相关类如Handler和Message类的源代码,同时使用他们来更新UI主线程的操作。因为Android的消息处理机制内容繁多,我们分为几部分来学习,大家可以关注这几讲内容,这一讲我们重点来学习一下Handler和Message.

一. Handler类介绍

1). 查看Android官网API Handler类

java.lang.Object
    android.os.Handler
Known Direct Subclasses
AsyncQueryHandler, AsyncQueryHandler.WorkerHandler, HttpAuthHandler, SslErrorHandler

一个Handler会允许你发送和处理Message或者Runnable对象关联到一个线程的消息队列MessageQueue中,每一个Handler的实例都会关联一个单一的线程和那个线程的消息队列中。当你创建一个一个新的Handler,它会绑定到你创建的线程和这个线程消息队列中。并且指向好它,它会让消息传递到关联好它的消息队列中,当它从消息队列出队的时候执行它。这里他们的如何关联的不是很懂!

对于Handler来说有两种主要的方式: 1. 计划好消息和Runnable将来的某一个时间点来执行它 2. 从一个不同的线程中执行Handler的入队操作。分发消息由下面的几个方法完成:

1) post(Runnable),
   2) postAtTime(Runnable, long), 
   3) postDelayed(Runnable, long), 
   4) sendEmptyMessage(int), 
   5) sendMessage(Message), 
   6) sendMessageAtTime(Message, long), 
   7) sendMessageDelayed(Message, long)

post方式的方法可以将一个Runable对象排列到消息队列中。sendMessage方式的方法可以通过 Handler的handleMessage(Message) 方法携带有bundle类型的数据的Message对象到队列中(需要你实现Handler的子类)。你可以通过上诉两种方式来出来Handler,你可以允许你的消息在消息队列中准备好就马上被处理,也可以处理之前指定一些延时让你实现超时或者基于时间的行为。
   当你的应用程序的进程被创建的时候,它的主线程专门用来处理正常运行的主线程的消息队列,(也就是说UI主线程有自己的消息队列,所以我们没必要在UI主线程中处理自己的消息)它关心的是管理顶层的应用对象(activities, broadcast receivers, etc)和他们创建的窗口。你可以创建你自己的线程,然后通过Handler与主线程沟通。就像上述说的通过post和sendMessage的方式,Runnable和Message会被计划的执行在Handler的消息队列中适时的进行处理。

二. Message类介绍

1). 查看Android官网API Message类

Java.lang.Object
       android.os.Message

定义一个message包含描述信息和任意的数据对象发送给Handler。这个对象包含两个额外的int类型的属性和一个Object类型的属性,它可以让你不需要去做一些强制类型的转换的操作。如下图所示:

1) arg1 和 arg2 都是Message自带的用来传递一些轻量级存储int类型的数据,比如进度条的数据等。通过这个数据是通过Bundle的方式来转载的,读者可以自己查阅源代码研究。

2) obj 是Message自带的Object类型对象,用来传递一些对象。兼容性最高避免对齐进行类型转换等。

3) replyTo 是作为线程通信的时候使用.

4) what 用户自定义的消息码让接受者识别消息种类,int类型。

【注意】: 获得Message的构造方法最好的方式是调用Message.obtain() 和 Handler.obtainMessage()方法。以便能够更好被回收池所回收[这里读者可以研究一下obtain()的源代码即可明白]。而不是直接用 new Message的方式来获得Message对象。

三. 程序Demo

1. 实现通过 Thread + Handler + Message 的方式下载网络数据。程序结构如下图所示

2. 在Manifest.xml中添加网络权限,这里不再贴出,读者可以参考上面一讲内容

3. 布局文件中 activity_main.xml 中定义Button和ImageView控件,这里不再贴出,读者可以自己下载源代码查看

4. MainActivity.java 程序主代码

[java] view plain copy

  1. <span style="font-family:Courier New;">...</span>
  2. /**
  3. * 通过 Handler + Message 的方式下载网络数据
  4. * 通过子线程run()方法中下载数据,使用Message携带数据,然后用Handler发送消息并且处理消息来更新UI.
  5. *
  6. * @author AHuier
  7. */
  8. public class MainActivity extends Activity {
  9. private Button btn;
  10. private ImageView imageView;
  11. private String imgPath = "http://f.hiphotos.baidu.com/image/w%3D2048/sign=05793c21bba1cd1105b675208d2ac9fc/43a7d933c895d14350ee3c3272f082025aaf0703.jpg";
  12. private static final int DOWNLOAD_IMG = 1;
  13. private ProgressDialog dialog = null;
  14. private Handler handler = new Handler() {
  15. // 处理子线程给我们发送的消息。
  16. @Override
  17. public void handleMessage(android.os.Message msg) {
  18. byte[] data = (byte[])msg.obj;
  19. Bitmap bitmap = BitmapFactory.decodeByteArray(data, 0, data.length);
  20. imageView.setImageBitmap(bitmap);
  21. if(msg.what == DOWNLOAD_IMG){
  22. dialog.dismiss();
  23. }
  24. };
  25. };
  26. @Override
  27. protected void onCreate(Bundle savedInstanceState) {
  28. super.onCreate(savedInstanceState);
  29. setContentView(R.layout.activity_main);
  30. initComponent();
  31. btn.setOnClickListener(new View.OnClickListener() {
  32. @Override
  33. public void onClick(View v) {
  34. // TODO Auto-generated method stub
  35. new Thread(new MyThread()).start();
  36. dialog.show();
  37. }
  38. });
  39. }
  40. @Override
  41. public boolean onCreateOptionsMenu(Menu menu) {
  42. // Inflate the menu; this adds items to the action bar if it is present.
  43. getMenuInflater().inflate(R.menu.main, menu);
  44. return true;
  45. }
  46. private void initComponent() {
  47. btn = (Button) this.findViewById(R.id.button1);
  48. imageView = (ImageView) this.findViewById(R.id.imageView1);
  49. dialog = new ProgressDialog(this);
  50. dialog.setTitle("提示");
  51. dialog.setMessage("正在下载,请稍后...");
  52. dialog.setCancelable(false);
  53. }
  54. // 使用Handler Message MessageQueue Looper等方式去访问网络资源的时候,我们必须要开启一个子线程
  55. public class MyThread implements Runnable{
  56. // 在run方法中完成网络耗时的操作
  57. @Override
  58. public void run() {
  59. HttpClient httpClient = new DefaultHttpClient();
  60. HttpGet httpGet = new HttpGet(imgPath);
  61. HttpResponse httpResponse = null;
  62. try {
  63. httpResponse = httpClient.execute(httpGet);
  64. if(200 == httpResponse.getStatusLine().getStatusCode()){
  65. byte[] data = EntityUtils.toByteArray(httpResponse.getEntity());
  66. // 这里的数据data我们必须发送给UI的主线程,所以我们通过Message的方式来做桥梁。
  67. Message message = Message.obtain();
  68. message.obj = data;
  69. message.what = DOWNLOAD_IMG;
  70. handler.sendMessage(message);
  71. }
  72. } catch (Exception e) {
  73. // TODO: handle exception
  74. }
  75. }
  76. }
  77. }

5. 程序执行结果

四、程序Demo总结

1. 上述Demo中的Handler写法我们只是按图索骥的方式实现比较标准的写法,如果考虑android内存机制的情况下,private Handler的方式定义成为静态的会更好。在Android源码中Handler一般是定义成 protect 权限的。

  2.Handler主要是用来负责发送消息和处理消息的。

3.基于处理,其实这里面蕴含着一个消息队列的概念,这里为什么我们获得消息需要用到Obtain()的方式,而不是通过new的方式构建一个消息,这个问题下一讲我们会通过剖析Message中Obtain()的源代码来讨论。

更多关于Android消息处理机制请点击以下相关链接:

1. Android 消息处理机制之二: Message中obtain()源代码剖析

2. Android 消息处理机制之三: Handler中sendMessage()源代码剖析

源码下载地址:HandlerMessageTest

原文转自:http://blog.csdn.net/ahuier/article/details/17012005

原作者为 AHuier. 请尊重原作者版权

时间: 10-02

【Android 开发】: Android 消息处理机制之一: Handler 与 Message的相关文章

Android多线程----异步消息处理机制之Handler详解

关于Android的多线程知识,请参考本人之前的一篇博客:Android 多线程----AsyncTask异步任务详解 在Android当中,提供了异步消息处理机制的两种方式来解决线程之间的通信问题,一种是今天要讲的Handler的机制,还有一种就是之前讲过的 AsyncTask 机制. 一.handler的引入: 我们都知道,Android UI是线程不安全的,如果在子线程中尝试进行UI操作,程序就有可能会崩溃.相信大家在日常的工作当中都会经常遇到这个问题,解决的方案应该也是早已烂熟于心,即创

Android的消息处理机制——Looper,Handler和Message浅析

题外话: 说来有些惭愧,对于这三者的初步认识居然是在背面试题的时候.那个时候自己接触Android的时间还不长,学习的书籍也就是比较适合入门的<疯狂Android讲义>,当然在学到Handler这一部分的时候,书中也是有提到一些简单示例,后来在工作中需要用到这个MessageQueue的时候才开始真正琢磨了一下这三者的联系.如果想要对这三者好好理解一番,个人还是比较推荐<深入理解Android卷Ⅰ>.以下对这三者之间的恩怨纠葛的介绍和分析也是参考这本书的相关章节,算是一篇读书笔记吧

Android开发学习之路-使用Handler和Message更新UI

在Android中,在非主线程中更新UI控件是不安全的,app在运行时会直接Crash,所以当我们需要在非主线程中更新UI控件,那么就需要用到Handler和Message来实现 Demo中,使用到一个按钮和一个TextView,点击按钮之后改变TextView的内容,按钮点击时候新建一个进程,在进程中对UI控件进行修改. 1 public class MainActivity extends Activity implements OnClickListener { 2 private sta

从Handler+Message+Looper源代码带你分析Android系统的消息处理机制

PS一句:不得不说CSDN同步做的非常烂.还得我花了近1个小时恢复这篇博客. 引言 [转载请注明出处:http://blog.csdn.net/feiduclear_up CSDN 废墟的树] 作为Android开发人员,相信非常多人都使用过Android的Handler类来处理异步任务. 那么Handler类是怎么构成一个异步任务处理机制的呢?这篇 博客带你从源代码分析Android的消息循环处理机制.便于深入的理解. 这里不得不从"一个Bug引发的思考"開始研究Android的消息

从Handler+Message+Looper源码带你分析Android系统的消息处理机制

引言 [转载请注明出处:从Handler+Message+Looper源码带你分析Android系统的消息处理机制 CSDN 废墟的树] 作为Android开发者,相信很多人都使用过Android的Handler类来处理异步任务.那么Handler类是怎么构成一个异步任务处理机制的呢?这篇 博客带你从源码分析Android的消息循环处理机制,便于深入的理解. 这里不得不从"一个Bug引发的思考"开始研究Android的消息循环处理机制.说来话长,在某一次的项目中,原本打算开启一个工作线

Android 消息机制 (Handler、Message、Looper)

综合:http://blog.csdn.net/dadoneo/article/details/7667726 与 http://android.tgbus.com/Android/androidnews/201204/421642.shtml 一. 消息机制常用类的介绍和使用 在Android程序运行中,线程之间或者线程内部进行信息交互时经常会使用到消息,如果我们熟悉这些基础的东西及其内部的原理,将会使我们的Android开发变的容易.可以更好地架构系统.在学习Android消息机制之前,我们

Android 消息处理机制-Looper,Handler,MessageQueue

先膜拜下大神的帖子,从源码的角度分析android的Handler机制.   链接:android的消息处理机制(图+源码分析)——Looper,Handler,Message 这里就不赘言,仅仅做一些介绍,想看详细请猛戳上面的链接. android的消息机制由三块组成Looper,Handler,MessageQueue.我们知道在android中子线程中是不能够执行组件的更新操作的,既然这样我们只能在子线程中把消息发送给主线程,让主线程自己去更新,这里就会用到Handler.子线程通过调用主

Android应用程序消息处理机制(Looper、Handler)分析

文章转载至CSDN社区罗升阳的安卓之旅,原文地址:http://blog.csdn.net/luoshengyang/article/details/6817933 Android应用程序是通过消息来驱动的,系统为每一个应用程序维护一个消息队例,应用程序的主线程不断地从这个消息 队例中获取消息(Looper),然后对这些消息进行处理(Handler),这样就实现了通过消息来驱动应用程序的执行,本文将详细分析Android 应用程序的消息处理机制. 前面我们学习Android应用程序中的Activ

Android线程之异步消息处理机制(二)——Message、Handler、MessageQueue和Looper

异步消息处理机制解析 Android中的异步消息处理主要有四个部分组成,Message.Handler.MessageQueue和Looper. 1.Message Message是在线程之间传递的消息,它可以在内部携带少量的信息,用于在不同线程之间交换数据.上个例子中就使用了Message的what字段,除此之外还可以使用arg1和arg2字段来携带一些整形数据,使用obj字段携带一个Object对象. 2.Handler Handler顾名思义也就是处理者的意思,它主要是用于发送和处理消息的