背景
这几天在测试应用,所以为了看应用是否有内存泄露,就加入了LeakCanary,结果还真有,因为我们是直播应用,使用了声网的视频通话技术,所以发现退出的时候Activity没有被释放,并且打开几次就有几个实例,这就爽歪歪了。
- 先上内存泄露的图
从上图中可以看出,是在声网io.agora.rtc.internal.AudioRoutingController这里使用到了,所以从源码下手,先找到上边的类,然后在类里面搜索BluetoothHeadset,然后再查看当前类销毁的时候,有没有解绑BluetoothHeadset,源码如下
private void clearBTResource() {
if(this.mBTAdapter != null) {
this.mBTAdapter.closeProfileProxy(1, this.mBTHeadset);
this.mBTAdapter = null;
}
if(this.mBTHeadsetListener != null) {
this.mBTHeadsetListener = null;
}
}
public void uninitialize() {
Logging.d("AudioRoute", "uninitialize");
try {
this.clearBTResource();
Context context = (Context)this.mContext.get();
if(context != null) {
if(this.mHeadsetReceiver != null && this.mHeadsetReceiver.getRegistered()) {
context.unregisterReceiver(this.mHeadsetReceiver);
this.mHeadsetReceiver.setRegistered(false);
}
if(this.mBTHeadsetReceiver != null && this.mBTHeadsetReceiver.getRegistered()) {
context.unregisterReceiver(this.mBTHeadsetReceiver);
this.mBTHeadsetReceiver.setRegistered(false);
}
}
this.mHeadsetReceiver = null;
this.mBTHeadsetReceiver = null;
} catch (Exception var2) {
Logging.e("AudioRoute", "AudioRoutingController uninitialize fail: ", var2);
}
}
可以通过上边的源码看出,引用context的时候,作者已经使用软引用或者弱引用了(Context context = (Context)this.mContext.get();根据这个就可以看出),
然后跟踪mConext看到作者声明的时候直接使用了private WeakReference mContext;,
那么已经使用WeakReference来包装Context了,那么为何还会造成泄露呢?为什么还是回收不了呢?原因暂时没有找到,先说解决方法
解决
既然是使用Context造成的内存泄露,那么我们首先可以使用全局的Context来替代Activity的Context,这样整个应用也就一个实例,所以就可以解决此泄露问题。
存疑
我们都知道Context如果造成内存泄露,有两种解决方法,第一种是使用全局的来替代,第二种就是使用weakreference来解决,目前来看第二种不靠谱啊。不知有没有高手解释一下