Android面试集

Handler相关

涉及到的主要的类:HandlerLooperMessageQueueMessage
  • Handler:作用是将Message对象发送到MessageQueue中,同时将自己的引用复制给Message#target

  • Looper作用是将Message对象从MessageQueue中取出来,将其交给Handler#dispatchMessage(Message)方法。

  • MessageQueue作用是负责插入和取出MessageMessageQueue是有序的队列,它的内部是基于Message#when字段进行一个排序,由MessageQueue#enqueueMessage(Message,Long)方法设置,该字段表示一个相对时间,用于Message期望被分发的时间,该值是SystemClock#uptimeMillis()delayMillis之和,SystemClock#uptimeMillis()代表的是 ==自系统启动开始从0到调用改分发时相差的毫秒数==。

  • Message作用是作为一个信息载体。


Q:Message#when为什么不用System.currentTimeMillis()来表示

System.currentTimeMillis()代表的是从1970-01-01 00:00:00到当前时间的毫秒数,这个值是一个强关联系统时间的值,我们可以通过修改系统时间达到修改该值的目的,所以该值是不可靠的值。

比如手机长时间没有开机,开机后系统时间重置为出厂时设置的时间,中间我们发送了一个延迟消息,过了一段时间通过NTP同步了最新时间,那么就会导致延迟消息失效,同时Message#when只是用时间差来表示先后关系,所以只需要一个相对时间就可以达成目的。


Q:子线程中可以创建Handler对象吗?

可以在子线程中创建,但是不可以在子线程中直接调用Handler的无参构造方法,因为Handler在创建时必须要绑定一个Looper对象,有两种方法绑定:

  • 先调用Looper.prepare()在当前线程初始化一个Looper
Looper.prepare();
Handler handler = new Handler();
// ....
// 这一步可别可少了
Looper.loop();
  • 通过构造方法传入一个Looper
Looper looper = .....;
Handler handler = new Handler(looper);

Q:Handler是如何与Looper关联的

  • 构造函数传入进行关联
  • 创建Handler如果是调用的无参构造的话,Handler里面会调用Looper的静态方法去获取looper进行绑定关联
public Handler(Callback callback, boolean async) {
   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 = callback;
   mAsynchronous = async;
}

Q:Looper是如何与Thread关联的

LooperThread之间是通过ThreadLocal关联的,这个可以看Looper#prepare()方法

private static void prepare(boolean quitAllowed) {
   if (sThreadLocal.get() != null) {
       throw new RuntimeException("Only one Looper may be created per thread");
   }
   sThreadLocal.set(new Looper(quitAllowed));
}

Looper中有一个ThreadLocal类型的sThreadLocal静态字段,Looper通过它的getset方法来赋值和取值。

由于ThreadLocal是与线程绑定的,所以我们只要把LooperThreadLocal绑定了,那LooperThread也就关联上了。


Q:ThreadLocal是什么

ThreadLocal是一个线程内部的数据存储类,通过它可以在指定的线程中存储数据,数据存储以后,只有在指定线程中可以获取到存储的数据,对于其他线程来说则无法获取到数据。


Q:Looper.loop()会退出吗

不会自动退出,但是我们可以通过Looper#quit()或者Looper#quitSafely()让它退出。

两个方法都是调用了MessageQueue#quit(boolean)方法,当MessageQueue#next()方法发现已经调用过MessageQueue#quit(boolean)时会return null结束当前调用,否则的话即使MessageQueue已经是空的了也会阻塞等待。


Q:MessageQueue#next 在没有消息的时候会阻塞,如何恢复

当其他线程调用MessageQueue#enqueueMessage时会唤醒MessageQueue,这个方法会被Handler#sendMessageHandler#post等一系列发送消息的方法调用。

boolean enqueueMessage(Message msg, long when) {
   // ...
   synchronized (this) {
       // ...
       boolean needWake;
       if (p == null || when == 0 || when < p.when) {
           // New head, wake up the event queue if blocked.
           msg.next = p;
           mMessages = msg;
           needWake = mBlocked;
       } else {
           // ...
       }
       if (needWake) {
           nativeWake(mPtr); // 唤醒MessageQueue
       }
   }
   return true;
}

Q:Looper.loop()方法是一个死循环,为什么不会阻塞APP

如果说操作系统是由中断驱动的,那么Android的应用在宏观上可以说是Handler机制驱动的,所以主线程中的Looper不会一直阻塞的

  • 当队列中只有延迟消息的时候,阻塞的时间等于头结点的when减去当前时间,时间到了以后会自动唤醒。
  • Android中一个进程中不会只有一个线程,由于Handler的机制,导致我们如果要操作View等都要通过Handler将事件发送到主线程中去,所以会唤醒阻塞。
  • 传感器的事件,如:触摸事件、键盘输入等。
  • 绘制事件:我们知道要想显示流畅那么屏幕必须保持60fps的刷新率,那绘制事件在入队列时也会唤醒。
  • 总是有Message源源不断的被加入到MessageQueue中去。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,029评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,238评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,576评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,214评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,324评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,392评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,416评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,196评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,631评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,919评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,090评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,767评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,410评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,090评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,328评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,952评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,979评论 2 351

推荐阅读更多精彩内容