handler looper 和 线程

 

 Handler的概念

  顾名思义,handler在英语中是“操作着,处理者的意思”,而官方的文档给出的概念是,handler允许你发送或者处理Message对象或者Runable对象,这两个对象都是与线程的Message queue相关联的。每一个handler的实例(一个线程中可以有多个)都与单个的线程和那个线程对应的Messagequeue 关联,而处理的先后则按照发送消息的先后,先进先出进行处理。

根据自己的理解,handler主要负责message的发送与消息的处理。

 

如下面例子     Handler handler=new Handler();//负责允许runable对象

	Handler handler1 = new Handler()//负责接受message对象
	{
		public void handleMessage(Message msg) {
      
			System.out.println("hanler1:run"+Thread.currentThread().getId()); 
        }
	};
Runnable r=new Runnable(){ //这个runable对象,将在别的线程中执行 @Override public void run() { Message m = handler1.obtainMessage(); handler.post(new Runnable(){ @Override public void run() { // TODO Auto-generated method stub System.out.println("handler2"+Thread.currentThread().getId()); }}); m.sendToTarget(); } };

  

Thread t=new Thread (r); //执行上面的runable
t.start();

 

 我们来先看看结果:

从上面结果来看,可以得出一点结论,那就是不管是Message还是runable对象,虽然是在别的线程通过handler发送消息,但是handler接到消息后,处理过程还是在handler所在的那个线程中(也就是本例中的主线程中)

 

在网上有些文章说,Message和runable是又两个队列来管理的,其实不是,我们不妨将

m.sendToTarget();移到
handler.post的前面
Runnable r=new Runnable(){ 
        @Override 
        public void run() { 
        Message m = handler1.obtainMessage();
        m.sendToTarget();//这个移到前面
        handler.post(new Runnable(){ 
             
            @Override 
            public void run() { 
            // TODO Auto-generated method stub 
            System.out.println("handler    threadid="+Thread.currentThread().getId()); 
             
            }});
            
        } 
        
    }; 

 

 

 

 

 发现结果:

hanler1先执行了,说明,他们共用的一个队列。

实际上Looper通过从Message queue从取出一个message,然后由优先级的从高到底判断

message的类型,有三种类型(     1) Message里面的Callback,一个实现了Runnable接口的对象,其中run函数做处理工作;
                    2) Handler里面的mCallback指向的一个实现了Callback接口的对象,由其handleMessage进行处理;
                    3) 处理消息Handler对象对应的类继承并实现了其中handleMessage函数,通过这个实现的handleMessage函数处理消息。)

 

 

Looper的概念

根据官方文档的概念(Class used to run a message loop for a thread.)用来为线程处理消息循环

而线程中默认是没有Looper的。为了给一个线程创建looper,执行Looper.prepar() 和Looper.loop()

实现如下

class LooperThread extends Thread {
      public Handler mHandler;

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

          mHandler = new Handler() {
              public void handleMessage(Message msg) {
                  // process incoming messages here
              }
          };

          Looper.loop();
      }
  }

 

Looper在执行loop()里面处理消息

有人说为啥主线程里面没有Looper.prepare() 和Looper.loop()呢,其实系统已经为主线程默认添加了这两个方法,所以在主线程中,我们可以直接初始化handler。

而且如果你要自定义实现looper之后,必须在线程退出的时候,调用Looper.quit()。

 

下面再总结下:

每一个线程都可以包含一个Looper和一个消息队列

 

在android中ui线程(也即主线程)默认有一个Looper和消息队列

 

并且每一个Handler都有一个对应得message queue,而且一个线程中只有一个message queue,通过handler向message queue 发送消息,有两种方式发送sendMessage,和post

两者都可以更新UI线程。

 

经过测试,handler在哪个线程里面,不管是发送给他的message还是runale都在相对应的那个handler线程里面执行(一般是 主线程)

http://mobile.51cto.com/abased-375243.htm这篇文章上面说的有问题

posted @ 2014-12-07 16:50  雪狼的程序故事  阅读(4533)  评论(1编辑  收藏  举报