5分钟完全理解android handler

Handler机制简介

Handler是android中最重要组成部分之一,Handler机制可以看做是一个消息阻塞队列,APP启动后很快就进入死循环(while循环),不断的读取消息队列中的消息,每个线程最多只有一个消息队列,没有消息时就阻塞,有就立马执行。所有消息排队执行,因为是一个线程,所以同时只能执行一个消息。android的view绘制,事件响应(点击,触摸屏幕等)都是把消息发送到了主线程的消息队列,等待android APP的执行(这点可以通过手动抛出异常查看错误堆栈来验证)。包括自己在主线程new 的handler最终也是把消息插入到了主线程消息队列中。从以上来看android主线程大部分时间是空闲的。当点击屏幕后手机能立马响应也可以看出android主线程大部分时间是空闲的。虽然主线程要处理的事情狠多,很杂,很琐碎(view布局、绘制,事件分发等等),但处理时间都很短暂,可以保证很快处理完毕,然后等待下一个消息的到来。android handler机制简可以实现所有view相关的操作都在主线程进行,从而避免了使用 锁 。具体实现代码 如下。

java工程实现Handler机制代码

下面的代码跟android的handler机制主要原理完全一致,但不依赖android系统。


import com.handler.Handler;
import com.handler.Looper;
import com.handler.Message;

public class Main {

    public static void main(String[] args) {

        new Main().start();
        
    }
    
    private void start(){
        //创建该线程唯一的消息队列,线程安全的阻塞队列
        Looper.prepare();

        onCreate();
        
        //死循环,阻塞式,执行下面代码后主线程就会去获取消息队列里的消息,没有消息时就阻塞,有就执行。执行Looper.loop前即使消息队列里有消息,消息也不会执行,因为主线程还没有去检查消息队列。
        Looper.loop();
        
        //下面 的代码通常不会执行,除非手动让主线程消息队列退出。退出主线程消息队列后android的view布局、绘制,事件分发就不执行了,所以android APP也没必要继续执行了,所以android采用了抛出异常的方式结束APP。
        System.out.println("exit........");
        throw new RuntimeException("Main thread loop unexpectedly exited");

    }
    private void onCreate() {
        //////////////////////////////////////////////////////////
        ////// 下面的操作相当于运行在android的UI线程中 ////////////
        //////////////////////////////////////////////////////////

        final Thread thread = Thread.currentThread();
        System.out.println("main thread=" + thread);

        Handler handler = new Handler() {
            @Override
            public void handleMessage(Message msg) {
                //若thread == Thread.currentThread(),则证明已经运行在主线程中了
                System.out.println("current thread is main thread? " + (thread == Thread.currentThread()));
                System.out.println(msg);
                System.out.println();
            }
        };
        // 测试1       主线程创建handler,子线程使用该handler发送消息 
        new Thread() {
            public void run() {
                try {//模拟耗时操作
                    Thread.sleep(1000 * 2);
                } catch (InterruptedException e) {
                }
                Message message = new Message();
                message.obj = "new Thread" + Thread.currentThread();
                message.what = (int) System.currentTimeMillis();
                //在子线程中发送消息 
                handler.sendMessage(message);
                
                try {
                    Thread.sleep(1000 * 2);
                } catch (InterruptedException e) {
                }
                
                message = new Message();
                message.obj = "hanler...waht==1" ;
                message.what = 1;
                //在子线程中发送消息 
                handler.sendMessage(message);
                

                message = new Message();
                message.obj = "hanler...waht==2" ;
                message.what = 2;
                //在子线程中发送消息 
                handler.sendMessage(message);
                
                message = new Message();
                message.obj = "hanler...waht==3" ;
                message.what = 3;
                //在子线程中发送消息 
                handler.sendMessage(message);
                
            };
        }.start();

        // 测试2 在thread内部创建handler,结果会抛出异常
        new Thread() {
            public void run() {
                try {
                    sleep(1000 * 3);
                } catch (InterruptedException e) {
                }
                /*
                 * 在线程内部使用默认构造函数创建handler会抛出异常。
                 * android中也可以在子线程中创建Handler,但要在初始化时传入Looper,
                 * Looper.getMainLooper()获取到的就是主线程的Looper,所以可以这样创建
                 * 
                 * new Handler(Looper.getMainLooper()){
                        @Override
                        public void handleMessage(Message msg) {
                            //运行在主线程中
                        }
                    };
                 */
                Handler h = new Handler() {
                    public void handleMessage(Message msg) {

                        System.out.println("haneler msg...." + msg);
                    };
                };

                Message message = new Message();
                message.obj = "handler in new Thread";
                message.what = (int) System.currentTimeMillis();
                //在子线程中发送消息 
                h.sendMessage(message);

            };
        }.start();

        //////////////////////////////////////////////////////////
        ////// 上面的操作相当于运行在android的UI线程中 ////////////
        //////////////////////////////////////////////////////////

    }
}


运行结果


main thread=Thread[main,5,main]  
current thread is main thread? true  
what=18175614 obj=new ThreadThread[Thread-0,5,main]  
  
Exception in thread "Thread-1" java.lang.RuntimeException: Can't create handler inside thread that has not called Looper.prepare()  
    at com.handler.Handler.<init>(Handler.java:14)  
    at Main$3$1.<init>(Main.java:103)  
    at Main$3.run(Main.java:103)  
current thread is main thread? true  
what=1 obj=hanler...waht==1  
  
current thread is main thread? true  
what=2 obj=hanler...waht==2  
  
current thread is main thread? true  
what=3 obj=hanler...waht==3  

Handler代码


package com.handler;  
  
  
public class Handler {  
  
  
    private MessageQueue messageQueue;  
      
    public Handler() {  
  
        Looper looper=Looper.myLooper();  
          
        if (looper==null) {  
             throw new RuntimeException(  
                        "Can't create handler inside thread that has not called Looper.prepare()");  
                 
        }  
          
        this.messageQueue=looper.messageQueue;  
    }  
  
    public void sendMessage(Message msg) {  
          
        //Looper循环中发现message后,调用message.targer就得到了当前handler,使用taget.handleMessage  
        //就把消息转发给了发送message时的handler的handleMessage函数  
        msg.target=this;  
          
        messageQueue.enqueueMessage(msg);  
          
    }  
      
    public void handleMessage(Message msg) {  
    }  
}  

Looper代码


package com.handler;

public class Looper {


    private static final ThreadLocal<Looper> threadLocal=new ThreadLocal<>();
    /**
     * 存储Message的队列,阻塞式,没有消息则一直等待
     */
    final MessageQueue messageQueue;
    
    
    private Looper() {
        messageQueue=new MessageQueue();
    }

    /**为该线程创建Looper,
     * 若该线程已经有Looper了则不需要再次调用prepare
     */
    public  static void prepare() {
        if (threadLocal.get() != null) {
            throw new RuntimeException("Only one Looper may be created per thread");
        }
        threadLocal.set(new Looper() );
    }
    
    public static void loop() {
        Looper looper=myLooper();
        if (looper == null) {
            throw new RuntimeException("No Looper; Looper.prepare() wasn't called on this thread.");
        }
        MessageQueue messageQueue=looper.messageQueue;
        
        for(;;){
            Message message=messageQueue.next();
            message.target.handleMessage(message);
        }
    }
    
    /**
     * 获取当先线程的Looper
     * @return
     */
    public static Looper myLooper() {
        return threadLocal.get();
    }
}


MessageQueued代码


package com.handler;  
  
import java.util.concurrent.BlockingQueue;  
import java.util.concurrent.LinkedBlockingQueue;  
  
public class MessageQueue {  
  
  
    private BlockingQueue<Message>blockingQueue=new LinkedBlockingQueue<>();  
      
    /** 
     * 阻塞式,没有消息则一直等待 
     * @return 
     */  
    public Message next() {  
        try {  
            return blockingQueue.take();  
        } catch (InterruptedException e) {  
            throw new RuntimeException();  
        }  
    }  
      
    /** 
     * 插入到消息队列尾部 
     * @param message 
     */  
    void enqueueMessage(Message message) {  
        try {  
            blockingQueue.put(message);  
        } catch (InterruptedException e) {  
            // TODO Auto-generated catch block  
            e.printStackTrace();  
        }  
    }  
}  

ThreadLocal简单实现

ThreadLocal内部原理和下面实现方式不同,但达到的效果是相同的,本篇主要介绍Handler机制,简化了ThreadLocal


package com.handler;  
  
import java.util.HashMap;  
import java.util.Map;  
/** 
 * ThreadLocal简单实现 
 * @author Young 
 * 
 * @param <T> 
 */  
public class ThreadLocal<T> {  
  
  
    private Map<Thread,T>map;  
  
    public ThreadLocal() {  
        map=new HashMap<>();  
    }  
      
    public void set(T obj) {  
        map.put(Thread.currentThread(),obj);  
    }  
      
    public T get() {  
        return map.get(Thread.currentThread());  
    }  
      
}  

Message代码


package com.handler;  
  
public class Message {  
  
    Handler target;  
    public Object obj;  
    public int what;  
  
    @Override  
    public String toString() {  
        return   "what="+what+" obj="+obj.toString();  
    }  
      
}  


以上就是android Handler机制原理代码了。

android还提供了HandlerThread,其实是对Handler和Thread的封装。

先看一下HandlerThread使用方式


Handler myHandler;
 new HandlerThread("Compress-Thread") {  
            @Override  
            protected void onLooperPrepared() {  
                super.onLooperPrepared();  
                myHandler = new Handler();  
                myHandler.post(new Runnable(){
                   @Override  
                    public void run() {  
                                //在HandlerThread线程执行
                    } 
                    
                });
            }  
        }.start();

//不要在这使用myHandler发送消息,因为myHandler是在onLooperPrepared中创建的,onLooperPrepared又是运行在HandlerThread线程的,所以刚执行到这时HandlerThread线程可能还没有创建完,onLooperPrepared也就不会执行,myHandler自然是null

每次new HandlerThread都会创建一个新线程,当我们使用myHandler发送消息时消息就会在HandlerThread线程执行。HandlerThread线程内部也是一个死循环,通常不会退出,当通过myHandler为其发送消息时就会从阻塞中醒来执行消息,执行完消息队列里的消息后就又阻塞。HandlerThread可以排队执行消息,保证能按加入消息的先后顺序执行。比如我们需要压缩很多图片时,就可以使用HandlerThread,主线程直接把图片通过myHandler发送到HandlerThread,HandlerThread就可以执行图片压缩,所有压缩任务都按添加顺序依次执行。

来自我的博客

http://blog.csdn.net/qingchunweiliang/article/details/50448365

END

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 218,525评论 6 507
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,203评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 164,862评论 0 354
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,728评论 1 294
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,743评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,590评论 1 305
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,330评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,244评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,693评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,885评论 3 336
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,001评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,723评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,343评论 3 330
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,919评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,042评论 1 270
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,191评论 3 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,955评论 2 355

推荐阅读更多精彩内容