1.Handler是什么?为什么要使用Handler呢?
它是一个可以实现多线程间切换的类,那我们为什么要在使用多线程呢?因为Android中会有一些耗时操作,系统推荐将这些耗时的操作放到后台线程运行,否则可能造成Android的ANR(手机的无响应),但运行结束后得到相应的结果有时我们需要操作UI,而UI线程却只能在UI线程(主线程)进行操作,这个时候我们就需要从后台线程切换到UI线程,进行相应的UI更新,此时Handler就派上了用场。
2.主线程使用Handler
Handler的基本使用还是很简单的,如下图:
这两种方法会将消息加入消息队列,并返回一个boolean表示是否加入队列,当然可以是普通的加入(放到消息队列的尾部) 如:post/sendMessage,如果消息队列没有消息的话就会立即执行,也可以通过设置执行任务的延迟时间加入消息队列 如:postDelayed/sendMessageDelayed和postAtTime/sendMessageAtTime会在在延迟时间之后执行,并不一定会准时的执行,还可以将,当前的消息插入到队列的第一个 如:postAtFrontOfQueue/sendMessageAtFrontOfQueue。
虽然这两种都起到了同样的作用,但是当我们在观察源码的时候会发现他们的不一样。如图:dispatchMessage这个方法会在Looper的loop方法执行:msg.target.dispatchMessage(msg),当我们调用post时会把我们传过去的runnable赋值给Message的callback,而sendMessage则会执行else的代码,这里我们也可以看得出runnanle执行的优先级大于handlerMessage。
3.Message
我们无论以某一种方式使用Handler的时候,我们都会创建Message,只是post时系统帮我们创建了Message并最终调用sendMessage,那么我们就来介绍一下Message里面的一些我们用的到的一些属性:
当然还有一点需要我们注意就是创建Message时,我们也是两种方法创建它的,一是无脑创建new Message,然后设置各种你需要的参数,二是通过Message.obtain()获取消息。那么这两者的区别的是什么呢?哪个更好用呢?当前是推荐使用Message.obtain(),因为它允许我们在许多情况下避免分配新的对象,可以从消息池中取出一个Message拿来用。
4.后台线程使用Handler
有时候我们还会在后台线程使用Handler,这个时候如果直接使用的话通常会报错,如图:
系统会告诉我们你还没有调用过Looper.prepar()这个方法,它会帮我们为当前线程设置一个Looper,那我们来看看这个方法源码:
那我们可能会想那么系统是什么是怎么获取Looper的呢?它会在ActivityThread这个类中调用prepareMainLooper()中设置好主线程中的Looper对象,不过它不仅会设置好主线程的Looer,还会调用Looper.loop(),让它开启循环,不断的从消息队列中获取消息并交给Handler处理,如图(学习java的小伙伴是不是终于看到了传说中的main函数这个程序的主入口呢):
所以我们可以自己创建Looper并启动它就可以实现在后台线程使用Handler了,如图:
当然还有一个方法可以实现后台线程,那就是使用HandlerThread,它会自动帮我们创建好Message,我们在创建Handler的时候把已经创建好的Looper传过去就好了,使用方法如图:
使用HandlerThread时我们可能使用完了Handler,也就不需要Looper了,那么如何退出Looper呢?有种两种方法,但都是通过HandlerThread的实例去调用方法,一个是quit(),另一个是qiutSafely(),这两者还是有区别的,只是暂时我还没弄懂。
5.如何取消任务
这个其实很简单调用Handler的removeCallback(runnable)或removeMessage(what)就可以实现了,不过post对应removeCallback(runnable),sendMessage对应removeMessage(what)传消息类型就可以了。
6.Handler使用中的隐患
大家应该都会听说过Handler可能导致内存泄漏,那么是为什么呢?是因为Java中当我们使用普通内部类时,它会持有外部类的引用,哪怕没有明确的引用其实也会隐式的持有外部类的引用,如图:
这时候我们就可以分析得到,Handler引用着这个Activity,而Message又引用着Handler(因为Message中的target就是当前发消息的Handler),而MessageQueue又引用着Message且MessageQueue随着主线程的Looper会一直存在(因为当前是在主线程中使用Handler),哪怕这个Activity马上调用finish(),也不并不会被Java的垃圾回收机制回收,因为它还被别人引用着,这个时候我们需要想到如何解决它。
这个时候其实我们可以使用静态内部类去创建Handler,因为静态内部类并不用持有外部类的引用,所以我们也就不用担Activity不会被回收,但是如果我们还是需要在Handler中使用Activity呢?那么可以使用Java的弱引用,从而使得Activity可以被Java回收掉,使用方法如下:
还有一点我们可以进一步优化就是当Activity回调onDestroy时,我们可以Handler的removeMessages/removeCallback取消任务,这样就完美了,哈哈哈。
这是我第一次写技术文章,之前其实一直想写,但都偷懒了,希望能坚持下去,我希望可以站在初学者&自学者的角度把Android中的知识点很清楚的介绍给大家,希望大家喜欢(第一次写也还有很多不太会用的地方,设计到的代码都是贴的图片,希望可以理解)。
如果有错误希望指出来,有问题或者没看懂,都可以来问我的