2024年Android最新源码解析 Handler 面试宝典,面试阿里巴巴国际站运营会问什么问题

最后

对于程序员来说,要学习的知识内容、技术有太多太多,要想不被环境淘汰就只有不断提升自己,从来都是我们去适应环境,而不是环境来适应我们!

最后,我再重复一次,如果你想成为一个优秀的 Android 开发人员,请集中精力,对基础和重要的事情做深度研究

对于很多初中级Android工程师而言,想要提升技能,往往是自己摸索成长,不成体系的学习效果低效漫长且无助。整理的这些架构技术希望对Android开发的朋友们有所参考以及少走弯路,本文的重点是你有没有收获与成长,其余的都不重要,希望读者们能谨记这一点。

为了大家能够顺利进阶中高级、架构师,我特地为大家准备了一套高手学习的源码和框架视频等精品Android架构师教程,保证你学了以后保证薪资上升一个台阶。

以下是今天给大家分享的一些独家干货:

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化学习资料的朋友,可以戳这里获取

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

*   [答案](about:blank#_16)
  • 2、一个线程有几个Looper?如何保证

    • 考点

    • 答案

  • 3、Handler 内存泄漏原因?为什么其他的内部类没有说过这个问题

    • 考点

    • 答案

  • 4、为何主线程可以new Handler?如果想要在子线程中new Handler 要做些什么?

    • 考点

    • 答案

  • 5、子线程中维护的Looper,消息队列无消息的时候的处理方法是什么?有什么用?

    • 考点

    • 答案

  • 6、既然可以存在多个Handler 往MessageQueue 中添加数据(发消息时各个Handler 可能处于不同线程),那它内部是如何确保线程安全的?

    • 考点

    • 答案

  • 7、我们使用Message 时应该如何创建它?

    • 考点

    • 答案

  • 8、Looper 死循环为什么不会导致应用卡死

    • 考点

    • 答案

  • 总结

前言

================================================================

在Android 的中,高级面试中,我们经常会被问到Handler 相关的知识点,而且占重比例还比较大,这是什么呢?下面一起来看一张图:

其实一部手机,就是一个Handler,

由上图我们可以看出,整个APP 的启动流程: Launcher(APP): zygote -> jvm -> ActivityThread.main() ActivityThread.main() 就是我们APP 独有的main 启动方法,如图所示,绿色的部分,就是Handler 为我们开辟的独有空间,启动主线程独有的Looper,将当前App 独立出来。

由此我们可以得出一个结论:Handler 并不是只属于进程通讯,进程通讯只是Handler 的附属功能,而Handler 的真正功能是 所有的代码,都是在Handler 中运行的

1、一个线程有几个Handler

==============================================================================

考点


这里面试官其实想了解的是,你对Handler的认知,如果这个问题打不出来,那么面试官也不会再去问了。

答案


在说答案之前,先看一直 Handler 的执行流程图:

在这里插入图片描述

由上图,我们可以看出:

  • 一个线程里面,可以创建N个的Handler,如hander.sentXXX、 handler.postXX 都是在创建一个Handler。 每一个message,就是我们插入到消息队列 中的消息节点。

2、一个线程有几个Looper?如何保证

==================================================================================

考点


这里面试官其实想了解的是,你对Handler 流程及源码的理解。

答案


  • 在回答这个问题之前,我们再看一下Handler 的执行流程图:

    handler -> sendMessage -> messageQueue.enqueueMessage -> looper.loop() -> messasgeQueue.next() -> handler.dispatchMessage() -> handler.handerMessage() ,handler 发送message,进入messageQueue.enqueueMessage 队列,进入looper中经过loop 死循环的不断遍历,驱动队列一直前进,经过handler.dispatchMessage() 分发给handler.handerMessage,这样我们就走完了整个的Handler 流程,也可以直接看错,一个线程中,只有一个Looper

  • 如何保证的呢?ThreadLocal 多线程,线程上下文的存储变量,其实ThreadLocal 并不能存储任何的东西,但是在ThreadLocal 中,有一个ThreadLocalMap 集合,里面存储的<this, value> ,this 就是上下文,唯一的ThreadLocal key,key 唯一了,那么value 也就是唯一的,ThreadLocal在创建的时候,会有一个判断,如果已经创建了,会报异常,所以一个线程有唯一的ThreadLocal 就有唯一的looper。如下图:

ThreadLocal 线程隔离工具类

在这里插入图片描述

ThreadLocal 创建源码

在这里插入图片描述

3、Handler 内存泄漏原因?为什么其他的内部类没有说过这个问题

================================================================================================

考点


应该是考官想知道,你对于GC回收 JVM 相关的东西吧。

答案


在这里,我我们先看一段代码:


    Handler handler = new Handler(){

        @SuppressLint("HandlerLeak")

        @Override

        public void handleMessage(@NonNull Message msg) {

            Log.d("tiger", "handleMessage: ");

            View view = null;

            click(view);

            MainActivity2.this.click(view);

        }

    };



    public void click(View view) {

        

    }



  • 上面代码片段中的handler 代码,会标黄,并给予一个警告:这个处理程序类应该是静态的,否则可能发生内存泄漏。

  • 那么为什么其他类不会有呢?

    生命周期的问题。流程 sendMessage -> sendMessageAtTime -> enqueueMessage 在 enqueueMessage 中,有段代码 msg.target = this;,意思就是Message 会持有当前的handler,handler 已经成为了massage的一部分,假如我设置一个消息需要等待20分钟后执行,那么就意味,我的message会一直等待20分钟之后才会执行,message 持有 handler,handler 持有 (this)activity,这样就导致GC无法回收,JVM 通过可达性算法,告诉我们,没法到达,就无法回收。内部类的生命周期,一旦在外部类生命周期中被别的生命周期持有了,那么外部类也不能被释放。

4、为何主线程可以new Handler?如果想要在子线程中new Handler 要做些什么?

==============================================================================================================

考点


好烦啊,我也不知道啊,为什么还要要求这个?

答案


  • 在APP 启动的时候,就在ActivityThread.main() 方法中,就创建了looper.loop() ,源码如下:

    在这里插入图片描述

  • 那么如何在子线程中new Handler呢?只需要在子线程中,手动添加 Looper.prepare();Looper.loop(); 就可以了。请看下面代码


    public void click (View view){

            new Thread(new Runnable() {

                @Override

                public void run() {

                    Looper.prepare();

                    mHandler = new Handler() {

                        public void handleMessage(Message msg) {

                            // do something()

                        }

                    };

                    Looper.loop();

                }

            }).start();

        }



5、子线程中维护的Looper,消息队列无消息的时候的处理方法是什么?有什么用?

======================================================================================================

考点


对于源码的掌握程度

答案


  • 子线程中维护的Looper 在无消息的时候调用quit,可以结束循环。loop() 是一个死循环,想要退出,必须msg == null。请看下面源码:

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.recycleUnchecked();

        }

    }





  • 只有在调用quit 的时候,才会返回null。

Message next() {

        for (;;) {

            synchronized (this) {

                if (mQuitting) {

                    dispose();

                    return null;

                }

            }

        }

    }



void quit(boolean safe) {

        synchronized (this) {

            if (mQuitting) {

                return;

            }

            mQuitting = true;

        }

    }





  • 所以使用quit() 唤醒队列,执行loop() 退出循环,子线程looper 不在执行了。

在这里插入图片描述

  • 消息入队:根据时间排序,当队列满的时候,阻塞,直到用户通过next() 取出消息。当next 方法被调用的时候,通知MassageQueue 可以消息入队。

  • 消息出队:由Looper.loop()进行循环, 对queue 进行轮询操作,当消息达到执行时间就取出来,当MessageQueue 为空的时候,队列阻塞,等消息调用queue massage 的时候,通知队列,可以取出消息,停止阻塞。

  • handler 没有使用多线程中的阻塞队列 BlockQueue,因为主线程(系统)也在使用,如果使用BlockQueue 设置上限的话,系统可能会出现卡顿等情况。

在这里插入图片描述

  • 从上图中,我们看可以出,handler 是一个生产者 - 消费者的设计模式。下面我们来看一下,looper 中的两种循环阻塞方式:
  1. 执行时间阻塞(没有到执行时间),nativePollOnce(long ptr, int timeoutMillis) 执行阻塞操作,timeoutMillis 为 -1 表示无限等待,直到事件发生为止,如果为0,无需等待,立即执行。请看下面源码:

    Message next() {

        final long ptr = mPtr;



### 最后

感觉现在好多人都在说什么安卓快凉了,工作越来越难找了。又是说什么程序员中年危机啥的,为啥我这年近30的老农根本没有这种感觉,反倒觉得那些贩卖焦虑的都是瞎j8扯谈。当然,职业危机意识确实是要有的,但根本没到那种草木皆兵的地步好吗?

Android凉了都是弱者的借口和说辞。虽然 Android 没有前几年火热了,已经过去了会四大组件就能找到高薪职位的时代了。这只能说明 Android 中级以下的岗位饱和了,现在高级工程师还是比较缺少的,很多高级职位给的薪资真的特别高(钱多也不一定能找到合适的),所以努力让自己成为高级工程师才是最重要的。

所以,最后这里放上我耗时两个月,将自己8年Android开发的知识笔记整理成的Android开发者必知必会系统学习资料笔记,上述知识点在笔记中都有详细的解读,里面还包含了腾讯、字节跳动、阿里、百度2019-2021面试真题解析,并且把每个技术点整理成了视频和PDF(知识脉络 + 诸多细节)。

![](https://img-blog.csdnimg.cn/img_convert/01cc8de7dd8aa583c318426b949ac334.webp?x-oss-process=image/format,png)

以上全套学习笔记面试宝典,吃透一半保你可以吊打面试官,只有自己真正强大了,有核心竞争力,你才有拒绝offer的权力,所以,奋斗吧!骚年们!千里之行,始于足下。种下一颗树最好的时间是十年前,其次,就是现在。

最后,赠与大家一句诗,共勉!

> 不驰于空想,不骛于虚声。不忘初心,方得始终。



**网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。**

**[需要这份系统化学习资料的朋友,可以戳这里获取](https://bbs.csdn.net/topics/618156601)**

**一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!**

1面试真题解析,并且把每个技术点整理成了视频和PDF(知识脉络 + 诸多细节)。

[外链图片转存中...(img-CoiYqZqM-1715679031576)]

以上全套学习笔记面试宝典,吃透一半保你可以吊打面试官,只有自己真正强大了,有核心竞争力,你才有拒绝offer的权力,所以,奋斗吧!骚年们!千里之行,始于足下。种下一颗树最好的时间是十年前,其次,就是现在。

最后,赠与大家一句诗,共勉!

> 不驰于空想,不骛于虚声。不忘初心,方得始终。



**网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。**

**[需要这份系统化学习资料的朋友,可以戳这里获取](https://bbs.csdn.net/topics/618156601)**

**一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!**

  • 30
    点赞
  • 16
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值