最近,在BOSS直聘网站上面投了自己的简历,今天晚上突然那边来了面试电话,直接就是一轮虐。
首先简单的做了一下自我介绍,其实这里面也有讲究的,当时没注意到。面试官对你进行面试,肯定是看了一下自己的简历的。这时候自我介绍就最好不要说一些简历上面没提到的东西。因为自己是在HW公司搞Android 机顶盒应用层开发的,也没有接触到一些新技术框架,平时都是自己自学,看看博客,参照网上的一些大神的文章,简单的学习了解Volley、ButterKnife、OKHttp、Retofit、EventBus。
接下来就问一下Android 消息机制,面试官有提示到Handler、Looper、MessageQueue、Message。
其实Android中每一个Thread都跟着一个Looper,Looper可以帮助Thread维护一个消息队列,但是Looper和Handler没有什么关系,每一个Looper对象由包含一个MessageQueue,MessageQueue里面又饱含着Message对象,Message对象会持有一个Handler对象的引用。当我们在子线程里面新建Handler对象的时候,必须要调用Looper. prepare()方法来创建与该Handler线程所对应的Looper对象了(其底层是使用ThreadLocal也就是线程局部变量来存储Looper对象)。当然了在主线程里面创建Handler就不需要调用了,应为Activity启动的时候就已经在main方法里面调用了。当我们调用Handler的sendMessage(Message)发送消息的时候,其实是通过Looper对象里面的MessageQueue的enqueueMessage(Message)添加到了其阻塞队列里面去了。然后Looper.loop()方法会从阻塞队列里面取出Message对象,回调Handler的handleMessage方法。
延迟time时间发送消息:mHandler.postDelayed(runnable,time)
取消发送:mHandler.removeCallbacks(runnable)
参考博客:
1. 自定义Handler
2. Android异步消息机制
第二个问题是,事件分发机制
这个问题很尴尬,当时由于很紧张,竟然忘记了dispatchTouchEvent() 事件分发函数名叫啥。回答的不是很好,这个需要反思一下。
什么是事件?当我们点击屏幕的时候会产生一个Touch事件,这个事件被封装为一个名叫MotionEvent对象,有MotionEvent.ACTION_DOWN、MotionEvent.ACTION_MOVE、MotionEvent.ACTION_CANCLE(非人为操作)、MotionEvent.ACTION_UP。事件分发的本质其实就是将事件分发到具体的View处理的过程。首先事件会从Activity传到ViewGroup再到View。
dispatchTouchEvent() 事件分发函数
onInterceptTouchEvent(),只存在于ViewGroup类中,事件拦截函数
onTouchEvent() 事件处理函数
参考博客
1. View一问一答*事件分发机制
第三个问题就是Activity的启动模式
其实这个问题可以扩展出很多关于Activity的生命周期、切换屏幕、异常退出、数据保存、进程的优先级等等。可以参考博客BAT大牛面试之谈解密Android
- standard 这是默认模式,每次激活Activity时都会创建Activity实例,并放入任务栈中。使用场景:大多数Activity。
- singleTop 如果在任务的栈顶正好存在该Activity的实例,就重用该实例( 会调用实例的 onNewIntent() ),否则就会创建新的实例并放入栈顶,即使栈中已经存在该Activity的实例,只要不在栈顶,都会创建新的实例。使用场景如新闻类或者阅读类App的内容页面。
- singleTask 如果在栈中已经有该Activity的实例,就重用该实例(会调用实例的 onNewIntent() )。重用时,会让该实例回到栈顶,因此在它上面的实例将会被移出栈。如果栈中不存在该实例,将会创建新的实例放入栈中。使用场景如浏览器的主界面。不管从多少个应用启动浏览器,只会启动主界面一次,其余情况都会走onNewIntent,并且会清空主界面上面的其他页面。
- singleInstance 在一个新栈中创建该Activity的实例,并让多个应用共享该栈中的该Activity实例。一旦该模式的Activity实例已经存在于某个栈中,任何应用再激活该Activity时都会重用该栈中的实例( 会调用实例的 onNewIntent() )。其效果相当于多个应用共享一个应用,不管谁激活该 Activity 都会进入同一个应用中。使用场景如闹铃提醒,将闹铃提醒与闹铃设置分离。singleInstance不要用于中间页面,如果用于中间页面,跳转会有问题,比如:A -> B (singleInstance) -> C,完全退出后,在此启动,首先打开的是B。
第四个问题是说了一下平时都关注了哪些开源框架,对其了解程度。
- Volley 异步网络请求框架,适合数据量较小,但访问频繁的操作。其内部会有一个缓存线程CacheDispatcher、4个网络请求线程NetWorkDispatcher。
- ButterKnife 依赖注入框架,采用的是编译器时期注入方式,得到注解的字段属性与所在类 , 进而生成注入文件,生成一个注入类的内部类,再进行字段处理 , 编译之后就会合并到注入类中,达到植入新代码段的目的。
当然还有一些像OKHttp网络请求框架、Retorfit2、RxJava 响应式编程、EventBus事件总线。这些东西后面再打算进行详细分析。