Android中的消息机制

在分析Android消息机制之前。我们先来看一段代码:

public class MainActivity extends Activity implements View.OnClickListener {  

    private TextView stateText;
    private Button btn;  

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        stateText = (TextView) findViewById(R.id.tv);
        btn = (Button) findViewById(R.id.btn);  

        btn.setOnClickListener(this);
    }  

    @Override
    public void onClick(View v) {
        new WorkThread().start();
    }  

    //工作线程
    private class WorkThread extends Thread {
        @Override
        public void run() {
            //......处理比較耗时的操作  

            //处理完毕后改变状态
            stateText.setText("completed");
        }
    }
} 

这段代码似乎看上去非常正常,可是当你执行时就会发现。它会报一个致命性的异常:

ERROR/AndroidRuntime(421): FATAL EXCEPTION: Thread-8
ERROR/AndroidRuntime(421): android.view.ViewRoot$CalledFromWrongThreadException:
Only the original thread that created a view hierarchy can touch its views.

究竟是怎么回事呢?原因在于,Android系统中的视图组件并非线程安全的,假设要更新视图,必须在主线程中更新。不能够在子线程中运行更新的操作。

既然这样,我们就在子线程中通知主线程,让主线程做更新操作吧。那么,我们怎样通知主线程呢?我们须要使用到Handler对象。

我们略微改动一下上面的代码:

public class MainActivity extends Activity implements View.OnClickListener {

	private static final int COMPLETED = 0;

	private TextView stateText;
	private Button btn;

	private Handler handler = new Handler() {
		@Override
		public void handleMessage(Message msg) {
			if (msg.what == COMPLETED) {
				stateText.setText("completed");
			}
		}
	};

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        stateText = (TextView) findViewById(R.id.tv);
        btn = (Button) findViewById(R.id.btn);

        btn.setOnClickListener(this);
    }

	@Override
	public void onClick(View v) {
		new WorkThread().start();
	}

	//工作线程
	private class WorkThread extends Thread {
		@Override
		public void run() {
			//......处理比較耗时的操作

			//处理完毕后给handler发送消息
			Message msg = new Message();
			msg.what = COMPLETED;
			handler.sendMessage(msg);
		}
	}
}

通过上面这样的方式,我们就能够解决线程安全的问题,把复杂的任务处理工作交给子线程去完毕,然后子线程通过handler对象告知主线程,由主线程更新视图。这个过程中消息机制起着关键的数据。

以下,我们就来分析一下Android中的消息机制。

熟悉Windows编程的朋友知道Windows程序是消息驱动的,而且有全局的消息循环系统。

Google參考了Windows的消息循环机制,也在Android系统中实现了消息循环机制。Android通过Looper、Handler来实现消息循环机制。Android的消息循环是针对线程的。每一个线程都能够有自己的消息队列和消息循环。

Android系统中的Looper负责管理线程的消息队列和消息循环。通过Looper.myLooper()得到当前线程的Looper对象,通过Looper.getMainLooper()得到当前进程的主线程的Looper对象。

前面提到。Android的消息队列和消息循环都是针对详细线程的,一个线程能够存在一个消息队列和消息循环,特定线程的消息仅仅能分发给本线程。不能跨线程和跨进程通讯。可是创建的工作线程默认是没有消息队列和消息循环的。假设想让工作线程具有消息队列和消息循环,就须要在线程中先调用Looper.prepare()来创建消息队列,然后调用Looper.loop()进入消息循环。以下是我们创建的工作线程:

	class WorkThread extends Thread {
	      public Handler mHandler;

	      public void run() {
	          Looper.prepare();

	          mHandler = new Handler() {
	              public void handleMessage(Message msg) {
	                  // 处理收到的消息
	              }
	          };

	          Looper.loop();
	      }
	  }

这样一来。我们创建的工作线程就具有了消息处理机制了。

那么。为什么前边的演示样例中,我们怎么没有看到Looper.prepare()和Looper.loop()的调用呢?原因在于,我们的Activity是一个UI线程。执行在主线程中,Android系统会在Activity启动时为其创建一个消息队列和消息循环。

前面提到最多的是消息队列(MessageQueue)和消息循环(Looper)。可是我们看到每一个消息处理的地方都有Handler的存在,它是做什么的呢?Handler的作用是把消息增加特定的Looper所管理的消息队列中,并分发和处理该消息队列中的消息。构造Handler的时候能够指定一个Looper对象,假设不指定则利用当前线程的Looper对象创建。以下是Handler的两个构造方法:

/**
     * Default constructor associates this handler with the queue for the
     * current thread.
     *
     * If there isn‘t one, this handler won‘t be able to receive messages.
     */
    public Handler() {
        if (FIND_POTENTIAL_LEAKS) {
            final Class<? extends Handler> klass = getClass();
            if ((klass.isAnonymousClass() || klass.isMemberClass() || klass.isLocalClass()) &&
                    (klass.getModifiers() & Modifier.STATIC) == 0) {
                Log.w(TAG, "The following Handler class should be static or leaks might occur: " +
                    klass.getCanonicalName());
            }
        }

        mLooper = Looper.myLooper();
        if (mLooper == null) {
            throw new RuntimeException(
                "Can‘t create handler inside thread that has not called Looper.prepare()");
        }
        mQueue = mLooper.mQueue;
        mCallback = null;
    }

/**
     * Use the provided queue instead of the default one.
     */
    public Handler(Looper looper) {
        mLooper = looper;
        mQueue = looper.mQueue;
        mCallback = null;
    }

以下是消息机制中几个重要成员的关系图:

一个Activity中可以创建出多个工作线程,假设这些线程把他们消息放入Activity主线程的消息队列中,那么消息就会在主线程中处理了。

由于主线程一般负责视图组件的更新操作,对于不是线程安全的视图组件来说,这样的方式可以非常好的实现视图的更新。

那么,子线程怎样把消息放入主线程的消息队列中呢?仅仅要Handler对象以主线程的Looper创建,那么当调用Handler的sendMessage方法,系统就会把消息主线程的消息队列,而且将会在调用handleMessage方法时处理主线程消息队列中的消息。

对于子线程訪问主线程的Handler对象,你可能会问,多个子线程都訪问主线程的Handler对象,发送消息和处理消息的过程中会不会出现数据的不一致呢?答案是Handler对象不会出现故障,由于Handler对象管理的Looper对象是线程安全的,无论是加入消息到消息队列还是从消息队列中读取消息都是同步保护的,所以不会出现数据不一致现象。

时间: 2024-08-02 11:03:25

Android中的消息机制的相关文章

深入解析Android中Handler消息机制

Android提供了Handler 和 Looper 来满足线程间的通信.Handler先进先出原则.Looper类用来管理特定线程内对象之间的消息交换(MessageExchange).Handler消息机制可以说是Android系统中最重要部分之一,所以,本篇博客我们就来深入解析Android中Handler消息机制. Handler的简单使用 为什么系统不允许子线程更新UI 因为的UI控件不是线程安全的. 如果在多线程中并发访问可能会导致UI控件处于不可预期的状态,那为什么不对UI控件的访

浅析Android中的消息机制(转)

原博客地址:http://blog.csdn.net/liuhe688/article/details/6407225 在分析Android消息机制之前,我们先来看一段代码: 1 public class MainActivity extends Activity implements View.OnClickListener { 2 private TextView stateText; 3 private Button btn; 4 5 @Override 6 public void onC

浅析Android中的消息机制-解决:Only the original thread that created a view hierarchy can touch its views.

在分析Android消息机制之前,我们先来看一段代码: [html] view plaincopyprint? public class MainActivity extends Activity implements View.OnClickListener { private TextView stateText; private Button btn; @Override public void onCreate(Bundle savedInstanceState) { super.onC

浅析Android中的消息机制(转)

在分析Android消息机制之前,我们先来看一段代码: public class MainActivity extends Activity implements View.OnClickListener { private TextView stateText; private Button btn; @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); s

Android中对消息机制(Handler)的再次解读

今天遇到一些关于在子线程中操作Handler的问题,感觉又要研究源代码了,但是关于Handler的话,我之前研究过,可以参考这篇文章:http://blog.csdn.net/jiangwei0910410003/article/details/17021809.但是这篇文章没有说的那么深入了,所以这次就更深入的解读一下. 摘要 Android中的应用程序都是通过消息驱动的,系统为每一个应用程序维护一个消息队列(MessageQueue),应用程序的主线程不断的从这个消息队列中获取消息(Loop

浅析Android中的消息机制

Android系统中的视图组件并不是线程安全的,如果要更新视图,必须在主线程中更新,不可以在子线程中执行更新的操作. Android系统中的Looper负责管理线程的消息队列和消息循环.通过Looper.myLooper()得到当前线程的Looper对象,通过Looper.getMainLooper()得到当前进程的主线程的Looper对象. Android的消息队列和消息循环都是针对具体线程的,一个线程可以存在一个消息队列和消息循环,特定线程的消息只能分发给本线程,不能跨线程和跨进程通讯. H

谈谈我对Android中的消息机制的理解之Handler,Looper和MessageQueue的解释

Handler的作用是:发送Message和处理Message,handler发送的Message其实就是发送给自己的对象进行处理,所以谁发送就是谁处理,但是这个绝对有意义,以为这样我们就可以通过Handler将消息的处理从一个线程转到另一个线程了,这个Message几经转手之后,处理它的对象虽然是同一个,但是处理它的线程就变了,变成了创建Handler对象的线程,而不是产生Message对象的线程(当然,这个两个线程可能是一个,但是这样使用handler就是第二个目的了),使用handler的

Android Framework 分析---消息机制Native层

在Android的消息机制中,不仅提供了供Application 开发使用的java的消息循环.其实java的机制最终还是靠native来实现的.在native不仅提供一套消息传递和处理的机制,还提供了自定义文件描述符的I/O时间的监听机制.下面我们从具体代码中分析一下. Native层的关键类: Looper.cpp.该类中提供了pollOnce 和wake的休眠和唤醒机制.同时在构造函数中也创建 管道 并加入epoll的机制中,来监听其状态变化. Looper::Looper(bool al

android中的多线程机制

Google参考了Windows的消息处理机制,在Android系统中实现了一套类似的消息处理机制.学习Android的消息处理机制,有几个概念(类)必须了解: 1.       Message 消息,理解为线程间通讯的数据单元.例如后台线程在处理数据完毕后需要更新UI,则可发送一条包含更新信息的Message给UI线程. 2.       Message Queue 消息队列,用来存放通过Handler发布的消息,按照先进先出执行. 3.       Handler Handler是Messa