Android架构中添加AutoDispose解决RxJava内存泄漏问题
概述
在我的上一篇文章解决rxjava内存泄漏(前篇):rxlifecycle详解及原理分析 中,详细阐述了
如何通过使用 rxlifecycle 解决android开发中rxjava的可能会导致的内存泄漏问题; rxlifecycle 内部的实现原理;在文章的最后,我提到了 autodispose 这个库,这个库同样可以解决android生命周期导致的rxjava的内存泄漏情况。
但是不得不考虑的是,目前国内的android开发圈子中,rxlifecycle已经逐渐为人所熟知,包括著名的一些开源架构也都采用rxlifecycle替代手动的处理方式 。比如 jessyan 的 mvparms 。
随着越来越多的项目添加了rxlifecycle,我需要给出一个足够有说服力的理由,足够让工程师同僚们去尝试将autodispose替换rxlifecycle,或者让他们在新的项目中优先考虑使用autodispose。
我花了一些时间研究了autodispose,并且在公司的新项目中尝试使用autodispose,我尝试将自己的所得分享出来,希望能够抛砖引玉——如果看完之后,仍旧觉得 autodispose 的思想和设计没什么用,至少也让客官您斩掉这个念头,继续安稳地使用 rxlifecycle 。
本文的主要内容如下:
autodispose的基础使用 autodispose的基本原理 autodispose和rxlifecycle的区别 如何添加到目前的android项目中(以mvp架构为例) 小结基础使用
1、添加依赖
compile 'com.uber.autodispose:autodispose:x.y.z' compile 'com.uber.autodispose:autodispose-android-archcomponents:x.y.z'
x.y.z请参照官网的最新版本,写这篇文章时,最新版本为0.6.1,即:
compile 'com.uber.autodispose:autodispose:0.6.1' compile 'com.uber.autodispose:autodispose-android-archcomponents:0.6.1'
2、在你的代码中直接使用,比如:
//在activity中使用 myobservable .map(...) .subscribeon(...) .observeon(...) .as(autodispose.autodisposable(androidlifecyclescopeprovider.from(this)) .subscribe(s -> ...);
通过给observable(或者single、completable、flowable等响应式数据类型,本文皆以observable为例)添加这行代码:
as(autodispose.autodisposable(androidlifecyclescopeprovider.from(lifecycleowner))
该observable就会在activity的ondestory()生命周期时,自动解除订阅,以防止因生命周期组件的生命周期而导致的rxjava内存泄漏事件。
看到这里,对比rxlifecycle的代码使用方式:
myobservable .compose(rxlifecycle.bind(lifecycle)) .subscribe();
似乎都差不多,都是通过添加一行代码实现rxjava事件流与组件生命周期的绑定,那么替换rxlifecycle的意义何在?
先按住这个问题不表,我先简单阐述一下autodispose的原理。
基本原理
直接长篇大论针对源码逐行分析,无疑是愚蠢的——我无法保证读者能够耐住性子看完枯燥无味的源码分析,然后兴致勃勃去尝试这个库,这与我初衷不符。
抛开细节,直接阐述其设计思想。
首先前提是,您需要对google最新的lifecycle组件有一定的了解,
同时,如果您对rxlifecycle的原理也掌握的话,相信对于接下来的内容
我们首先思考三个问题:
as(autodispose.autodisposable(androidlifecyclescopeprovider.from(this))
this是一个什么样的参数? 如何实现生命周期的绑定? as方法执行后生成了一个什么?1、我传了一个什么参数?
这个this直观的讲,就是activity本身,当然它也可以是fragment,这个参数对象只有一个要求,就是必须实现lifecycleowner接口。
lifecycleowner接口是google android官方架构组件:lifecycle的一个重要组件,在v7包中,fragmentactivity和fragment都实现了这个接口,实现了这个接口的对象都拥有生命周期(lifecycle)。
这意味着,不仅是appcompatactiviy(fragmentactivity的子类)和fragment,只要是实现了lifecycleowner的类,都可以作为参数传给autodispose,用以控制observable和组件生命周期的绑定。
2、如何实现生命周期的绑定
参考rxlifecycle的原理:
1.在activity中,定义一个observable(subject),在不同的生命周期发射不同的事件;
2.通过compose操作符(内部实际上还是依赖takeuntil操作符),定义了上游数据,当其接收到subject的特定事件时,取消订阅;
3.subject的特定事件并非是activityevent,而是简单的boolean,它已经内部通过combinelast操作符进行了对应的转化。
autodispose获取了activity(lifecycleowner)对象,并定义了一个新的observable,在activity的不同生命周期中,发射对应的事件。
和rxlifecycle很类似的是,autodispose在被订阅时,获取到activity当前的生命周期,并找到对应需要结束订阅的生命周期事件:
private static final function default_corresponding_events = new function() { @override public lifecycle.event apply(lifecycle.event lastevent) throws exception { switch (lastevent) { case on_create: return lifecycle.event.on_destroy;//比如我在oncreate到onstart之间订阅,我会在on_destroy时结束订阅 case on_start: return lifecycle.event.on_stop; case on_resume: return lifecycle.event.on_pause; case on_pause: return lifecycle.event.on_stop; case on_stop://如果是onpause之后订阅,会抛出异常 case on_destroy: default: throw new lifecycleendedexception("lifecycle has ended! last event was " + lastevent); } } };
也就是说,在我们的observablea订阅时,就已经知道了自己在activity的哪个生命周期让autodispose内部自定义的observableb自动发射事件,observablea监听到这个事件时且未dispose,解除订阅避免内存泄漏。
毕竟内存泄漏是少数,更大的可能是observablea早就执行完任务dispose了,因此observableb实际上就是一个maybe,类似于
observablea.takeuntil( maybe< true > )
下面为核心代码:
public static maybe resolvescopefromlifecycle( observable lifecycle, final e endevent) { return lifecycle.skip(1) .map(new function() { @override public boolean apply(e e) throws exception { return e.equals(endevent);//是否是要解除订阅的生命周期事件 } }) .filter(identity_boolean_predicate)//筛选为true的数据,即筛选解除订阅的生命周期事件 .map(transform_to_end)//转换为lifecycleendnotification.instance,这个枚举只用来通知解除订阅 .firstelement();//返回值为maybe,因为可能不到对应生命周期,observablea就已经完成任务,oncomplete() -> dispose了 } private static final function transform_to_end = new function() { @override public lifecycleendnotification apply(object o) throws exception { return lifecycleendnotification.instance; } }; public enum lifecycleendnotification { instance }
3、as方法执行后生成了一个什么?
as方法内部生成了一个autodisposeconverter对象,类似于compose,不同的是,observable通过compose生成的对象还是observable,但as方法生成的则是一个新的对象:
public final r as(@nonnull observableconverter converter)
实际上,抛开复杂的细节,autodispose最终将原来的observable,生成了一个新的autodisposeobservable对象, 在执行订阅时,也生成了一个新的autodisposingobserverimpl对象,篇幅所限,不再细述。
autodispose和rxlifecycle的区别
从上面的原理来讲,似乎autodispose和rxlifecycle两者没什么区别,原理都极为相似。
事实确实如此,因为autodispose本身就是很大一部分借鉴了rxlifecycle,同时,rxlifecycle的作者daniel lew 对于 autodispose的开发也有很多帮助:
以下摘录于autodispose的官方文档:
special thanks go to dan lew (creator of rxlifecycle), who helped pioneer this area for rxjava in android and humored many of the discussions around lifecycle handling over the past couple years that we’ve learned from. much of the internal scope resolution mechanics of
autodispose are inspired by rxlifecycle.
那么,究竟是什么原因,让rxlifecycle的作者daniel lew 在他自己的文章中,罗列出rxlifecycle在开发中的窘境,并同时强烈推荐使用autodispose呢?
i’m going to keep maintaining rxlifecycle because people are still using it (including trello), but in the long term i’m pulling away from it. for those still wanting this sort of library, i would suggest people look into autodispose, since i think it is better architecturally than rxlifecycle.
(我将会继续维护rxlifecycle因为很多人都在使用它,但是我会渐渐放弃这个库,我建议大家可以参考autodispose,因为我认为它的设计更加优秀 )
压倒性优势?
我们来看看rxlifecycle的局限性:
1、需要继承父类(rxactivity / rxfragment等)
对于设计来讲,【组合】的灵活度大多数情况都优于【继承】,而rxlifecycle在父类中声明了一个publishsubject,用来发射生命周期事件,这是导致其局限性的原因之一。
2、如何处处绑定生命周期?
最简单的例子,我的recyclerview的adapter中订阅了observable,亦或者,在mvp的架构或者mvvm架构中,我的presenter或者我的viewmodel无法直接获取rxactivity的引用(作为view层,更应该抽象为一个接口与presenter进行交互)。
这意味着,想要进行observable的生命周期绑定,在recyclerview的adapter中,我必须要通过将activity作为依赖,注入到adapter中:
new listadapter(rxactivity activity);
而对于presenter,我需要对view抽象接口进行instanceof 的判断:
if (view instanceof rxactivity) { return bindtolifecycle((rxactivity) view); }
如何添加到目前的android项目中(以mvp架构为例)
autodispose正常直接在项目中使用没有什么问题,比如:
//在activity中使用 myobservable .as(autodispose.autodisposable(androidlifecyclescopeprovider.from(this)) .subscribe(s -> ...);
但是,在实际的生产环境中,我们更希望有一个良好的方式,能够达到统一管理observable绑定生命周期的效果,在此笔者以mvp的架构为例,抛砖引玉,希望能够获得大家的意见和建议。
1、封装util类
首先封装util类,将职责赋予rxlifecycleutils:
public class rxlifecycleutils { private rxlifecycleutils() { throw new illegalstateexception("can't instance the rxlifecycleutils"); } public static autodisposeconverter bindlifecycle(lifecycleowner lifecycleowner) { return autodispose.autodisposable( androidlifecyclescopeprovider.from(lifecycleowner) ); } }
2、面向lifecycleowner接口
现在,只要持有lifecycleowner对象,observable都可以通过rxlifecycleutils.bindlifecycle(lifecycleowner)进行绑定。
比如我们在baseactivity/basefragment中添加代码:
public abstract class baseactivity extends appcompatactivity implements iactivity { //...忽略其他细节 protected autodisposeconverter bindlifecycle() { return rxlifecycleutils.bindlifecycle(this); } }
这样,在任何baseactivity的实现类中,我们都可以通过下述代码实现observable的生命周期绑定:
myobservable .as(bindlifecycle()) .subscribe(s -> ...);
但是这样的行为意义不大,我们更希望在presenter中也能够像上述代码一样达到observable生命周期的绑定,但是不持有activity的对象的同时,避免rxlifecycle中尴尬的instanceof判断。
可行吗,可行。
3、使用google官方lifecycle组件
首先让我们的ipresenter接口实现lifecycleobserver接口:
public interface ipresenter extends lifecycleobserver { @onlifecycleevent(lifecycle.event.on_create) void oncreate(@notnull lifecycleowner owner); @onlifecycleevent(lifecycle.event.on_destroy) void ondestroy(@notnull lifecycleowner owner); @onlifecycleevent(lifecycle.event.on_any) void onlifecyclechanged(@notnull lifecycleowner owner, @notnull lifecycle.event event); }
然后在basepresenter中管理lifecycleowner:
public class basepresenter implements ipresenter { @getter protected v mrootview; @getter protected m mmodel; private lifecycleowner lifecycleowner; public basepresenter(v rootview, m model) { this.mrootview = rootview; this.mmodel = model; } protected autodisposeconverter bindlifecycle() { if (null == lifecycleowner) throw new nullpointerexception("lifecycleowner == null"); return rxlifecycleutils.bindlifecycle(lifecycleowner); public void oncreate(@notnull lifecycleowner owner) { this.lifecycleowner = owner; } public void ondestroy(@notnull lifecycleowner owner) { this.lifecycleowner = null; if (mmodel != null) { mmodel.ondestroy(); this.mmodel = null; } this.mrootview = null; } }
最后在activity中添加presenter为观察者,观察activity的生命周期
public abstract class baseactivity
extends appcompatactivity implements iactivity { @inject protected p presenter; @override protected void oncreate(@nullable bundle savedinstancestate) { super.oncreate(savedinstancestate); setcontentview(getlayoutid()); //...忽略其他代码,比如butterknife、dagger2等 lifecycle.addobserver(presenter); } protected
这样,我们即使在presenter中,也能任意使用myobservable.as(bindlifecycle()) 方法了,和rxlifecycle相比,更加灵活。
篇幅所限,仅以最简单的设计思路进行阐述,更多情况的使用请在项目中自行拓展。