Android异步消息机制之Handler

Android消息机制

android消息机制的核心类:

  • Looper
  • Handler
  • MessageQueue---单链表
  • Message
  • ThreadLocal<T>

Handler异步通信机制工作流程图

Handler.Looper.MessageQueue关系图:

这里写图片描述

Handler核心类:

  • Handler:

    • Looper(当前线程ThraedLocal<Looper>)
    • MessageQueue(即为Looper关联的MessageQueue)
    • Imessenger(实现进程之间的通讯)
  • handler发送消息:

    • send方案发送消息(最终将消息放入消息队列中,即queue.enqueue()):
    
        private boolean enqueueMessage(MessageQueue queue, Message msg, long uptimeMillis) {
            msg.target = this;
            if (mAsynchronous) {
                msg.setAsynchronous(true);
            }
            return queue.enqueueMessage(msg, uptimeMillis);
        }
    
    • post方法发送消息:(将runnable()封装成Message后,放入队列中)
    
         private static Message getPostMessage(Runnable r) {
            Message m = Message.obtain();
            m.callback = r;
            return m;
        }
    
  • handler处理消息:

这里写图片描述

    public void dispatchMessage(Message msg) {
            if (msg.callback != null) {
                handleCallback(msg);
            } else {
                if (mCallback != null) {
                    if (mCallback.handleMessage(msg)) {
                        return;
                    }
                }
                handleMessage(msg);
            }
        }

Looper核心类:

Looper关联MessageQueue,每个Looper对象中都有一个MessageQueue用于存储消息, 关联一个ThreadLocal<Looper>用于存储当前线程的Looper对象.

  • looper.prepare()

        private static void prepare(boolean quitAllowed) {
            if (sThreadLocal.get() != null) {
                throw new RuntimeException("Only one Looper may be created per thread");
            }
            //将looper存储于ThreadLocal中
            sThreadLocal.set(new Looper(quitAllowed));
        }
  • looper.loop()
        //核心就不断从队列中取出消息,并处理消息, 如果队列中没有消息looper出现阻塞.
        public static void loop() {
            //.....
            for (;;) {
                Message msg = queue.next(); // might block
                if (msg == null) {
                    // No message indicates that the message queue is quitting.
                    return;
                }
                //.....
                msg.target.dispatchMessage(msg);
                //.....
                msg.recycleUnchecked();
            }
        }
  • 将线程变为looper线程, 类似于HandlerThread:

        public class LooperThread extends Thread {
            @Override
            public void run() {
                // 将当前线程初始化为Looper线程
                Looper.prepare();
        
                // ...其他处理,如实例化handler
        
                // 开始循环处理消息队列
                Looper.loop();
            }
        }

MessageQueue:

MessageQueue内部并不是队列,而是通过单链表结构来维护消息列表,单链表在插入和删除上比较有优势.

  • 插入功能(enqueueMessage())

  • 取出功能(next())即链表的删除功能)

主线程的消息循环:

Android的主线程就是ActivityThread,入口方法main();通常在新打开一个APK界面时,ActivityManagerService会为该APK启动一个新的新建ActivityThread作为该APK的主线程。该主线程的main函数主要做了两件事:

  • 新建ActivityThread对象。
  • 使用主线程进入消息循环。

    public final class ActivityThread { 
            ...
            
        public static void main(String[] args) {
                    ...
            Looper.prepareMainLooper();
            
            ActivityThread thread = new ActivityThread();
            thread.attach(false);
                  ...
            Looper.loop();
            
                    ...
        }
    }
  • 主线程消息循环后,需要一个Handler和消息队列交互:

        public final class ActivityThread {
    
            ...
        
            final H mH = new H();
        
            ...
        
            private class H extends Handler {
                ... 
        
                public void handleMessage(Message msg) {
                    ...
        
                    switch (msg.what) {
                        case LAUNCH_ACTIVITY: {
                            Trace.traceBegin(Trace.TRACE_TAG_ACTIVITY_MANAGER, "activityStart");
                            ActivityClientRecord r = (ActivityClientRecord)msg.obj;
        
                            r.packageInfo = getPackageInfoNoCheck(
                                    r.activityInfo.applicationInfo, r.compatInfo);
                            handleLaunchActivity(r, null);
                            Trace.traceEnd(Trace.TRACE_TAG_ACTIVITY_MANAGER);
                        } break;
        
                        ...
                    }
                }
            }
        }
  • ActivityThread通过ApplicationThread和AMS进行进程间通讯,AMS以进程间通讯的方法完成ActivityThread的请求后回调ApplicationThread中的Binder方法,然后ApplicationThread会向H发送消息,H收到消息后会将ApplicationThread逻辑切换到ActivityThread中执行,即切换到主线程中执行.
        public final class ActivityThread {
            ...
        
            private class ApplicationThread extends ApplicationThreadNative {
                ...
        
                public final void scheduleLaunchActivity(Intent intent, IBinder token, int ident,
                        ActivityInfo info, Configuration curConfig, CompatibilityInfo compatInfo,
                        int procState, Bundle state, List<ResultInfo> pendingResults,
                        List<Intent> pendingNewIntents, boolean notResumed, boolean isForward,
                        String profileName, ParcelFileDescriptor profileFd, boolean autoStopProfiler) {
        
                    ...
        
                    sendMessage(H.LAUNCH_ACTIVITY, r);
                }
        
                ...
            }
        
            ...
        
            private void sendMessage(int what, Object obj) {
                sendMessage(what, obj, 0, 0, false);
            }
        
            private void sendMessage(int what, Object obj, int arg1, int arg2, boolean async) {
        
                Message msg = Message.obtain();
                msg.what = what;
                msg.obj = obj;
                msg.arg1 = arg1;
                msg.arg2 = arg2;
                if (async) {
                    msg.setAsynchronous(true);
                }
                mH.sendMessage(msg);
            }
        
            ...
        }

Handler内存泄露:

This Handler class should be static or leaks might occur 。

  • 确保class内部的handler不含有外部类的隐式引用 。 同一个线程下的handler共享一个looper对象,消息中保留了对handler的引用,只要有消息在队列中,那么handler便无法被回收,如果handler不是static那么使用Handler的Service和Activity就也无法被回收。这就可能导致内存泄露。
  • 官方推荐将handler设为static类,并在里面使用弱引用WeakReference

public static class ServiceHandler extends Handler {
        WeakReference<BaseServiceActivity> activityRef;

        public ServiceHandler(WeakReference<BaseServiceActivity> activityRef) {
            this.activityRef = activityRef;
        }


        @Override
        public void handleMessage(Message msg) {

            Device device = null;
            switch (msg.what) {
                case Constant.MSG_NOTIFY_SERVER_CONNECT: //连上服务器
//                    Toast.makeText(activityRef.get(), "Have connected the socket server!", Toast.LENGTH_SHORT).show();
                    break;
                case Constant.MSG_NOTIFY_SERVER_DISCONNECT: //与服务器断开
//                    Toast.makeText(activityRef.get(), "Have disconnected the socket server!", Toast.LENGTH_SHORT).show();
                    break;
                case Command.SMS_REPLY_COMMAND: //GSM回复
                case Constant.MSG_NOTIFY_DEVICE_REPLY: //GPRS回复
                    activityRef.get().mHandler.removeMessages(Command.TIME_OUT_CMD);
                    activityRef.get().performNextActionWithCorrespondingCommand((String) msg.obj);
                    break;
                case Constant.MSG_NOTIFY_DEVICE_ONLINE_STATUS:
                    device = (Device) msg.obj;
                    activityRef.get().notifyDeviceOnlineStatus(device);
                    break;
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,616评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,020评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,078评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,040评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,154评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,265评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,298评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,072评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,491评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,795评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,970评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,654评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,272评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,985评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,223评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,815评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,852评论 2 351

推荐阅读更多精彩内容