Handler碎碎念
碎碎念
内存泄露
一、发生在延时消息、子线程阻塞后发过来的消息、以及网络请求结果回来后的消息,此时如果activity退出了,就会出现内存泄露。
解决办法:
- 延时消息(handler.sendMessageDelayed)
handler.removeAllMessage();当activity销毁时,清空handler中所有的消息 - 子线程阻塞或者网络请求回来后的消息
这种情况下,就不能使用上边的方法来处理,因为当activity销毁的时候,可能消息还没有添加到消息队列中,所以不会被清空。
这里就需要先清空消息队列,然后将handler置空,在发送消息的时候,对handler进行判空,当handler为空时,不再添加消息。
二、在Activity中直接创建handler并进行消息处理,当activity销毁时,如果消息还没有处理完成,此时,handler会持有外部activity的引用,造成内存泄露。
解决办法:
- Handler使用静态内部类创建,且对外部activity的引用使用弱引用
子线程为什么不能创建handler
一、 子线程创建handler会发生什么
- 当我们在子线程直接 new Handler的时候,会抛出异常
handler在创建的时候,构造方法中会调用Can't create handler inside thread " + Thread.currentThread() + " that has not called Looper.prepare()
Looper.myLooper()
方法来获取当前Handler的looper对象,如果looper对象为空,则会抛出上述异常。
在Looper类中,有一个静态属性sThreadLocal
,这是一个ThreadLocal对象,在ThreadLocal中保存了线程和looper对象的关系,Looper.myLooper()方法就是调用了ThreadLocal对象的get方法。
当我们在new Handler之前没有调用Looper.prepare()
方法为当前线程创建一个Looper对象的时候,就会报错。
二、为什么主线程可以直接创建Handler
应用在启动时会调用ActivityThread
的main
方法,在main方法中会调用Looper.prepareMainLooper();
在这个方法里