LiveData+ViewModel+RxJava2+autoDisposable解决内存泄漏

0.前言

使用Rxjava的人越来越多,rxjava现在已经更新到rxjava3了,因其简洁效率高的特点备受程序猿喜爱。然而由rxjava处理不但引起的内存泄漏问题也不得不让人头疼,每次使用完都要手动取消掉订阅也很麻烦而且不一定执行。为了解决这个问题,由此诞生了rxlifecycle,使用rxlifecycle可以跟踪lifeOwnner的生命周期从而自动取消订阅,但是却有许多限制和问题,详情见 为什么不使用 RxLifecycle?,作者已经在文章里面说了。文章最后还亲自推荐使用AutoDisposable,连他本人都认为rxlifecycle不如AutoDisposable了。网上关于AutoDisposable和LiveData相关使用的文章有很多,但是却找不到LiveData结合AutoDisposable一起使用的文章,着实让人费解。很明显,liveData使得代码仅仅在Activity处于活动状态下执行,如果代码使用了Rxjava订阅处理数据(在网略请求中很常见,网略请求经常使用Retrofit+okhttp+Rxjava,详情见笔者另一篇文章 使用Kotlin+Rretrofit+rxjava+设计模式+MVP封装网络请求),这个时候如果不及时取消订阅,则会引起内存泄漏

1.导入

// ViewModel and LiveData
implementation "android.arch.lifecycle:extensions:1.1.1"
// alternatively, just ViewModel
implementation "android.arch.lifecycle:viewmodel:1.1.1"
// alternatively, just LiveData
implementation "android.arch.lifecycle:livedata:1.1.1"
//autodispose
implementation 'com.uber.autodispose:autodispose-android-archcomponents:1.0.0-RC2'
implementation 'com.uber.autodispose:autodispose:1.0.0-RC2'

2.使用

这里举个栗子:

  • ViewModel
package foolish

import android.arch.lifecycle.LifecycleOwner
import android.arch.lifecycle.MutableLiveData
import android.arch.lifecycle.Observer
import android.arch.lifecycle.ViewModel
import beans.response.TaskBean
import com.uber.autodispose.AutoDispose
import com.uber.autodispose.android.lifecycle.AndroidLifecycleScopeProvider
import comment.LogUit
import io.reactivex.Observable
import io.reactivex.android.schedulers.AndroidSchedulers
import io.reactivex.observers.DisposableObserver
import io.reactivex.schedulers.Schedulers

//TestModel作为MVP或MVVM中的Present,专门负责处理数据
class TestModel:ViewModel(){
    val taskLiveData= MutableLiveData<Observable<TaskBean>>()//ViewModel内部持有LiveData
    fun initData( lifecycleOwner: LifecycleOwner){
        //此处选择在TestModel内部处理LiveData(网上教程很多是把LiveData放在View中处理)
        taskLiveData.observe(lifecycleOwner,object : Observer<Observable<TaskBean>>{
            override fun onChanged(observable: Observable<TaskBean>?) {
                observable!!.subscribeOn(Schedulers.io())
                        .observeOn(AndroidSchedulers.mainThread())
                        .`as`(AutoDispose.autoDisposable(AndroidLifecycleScopeProvider.from(lifecycleOwner)))//使用AutoDispose绑定lifecycleOwner,自动取消订阅
                        .subscribe(object : DisposableObserver<TaskBean>() {
                            override fun onComplete() {

                            }

                            override fun onNext(t: TaskBean) {
                                LogUit.printI("TaskPresentPrint", "t=${t.toString()}")
                            }

                            override fun onError(e: Throwable) {
                                //处理网略错误
                            }

                        })
            }

        })
    }
}

 //网略请求获取数据
    fun requestTaskDetail(taskDetailToBean: TaskDetailToBean){
        LogUit.printI(TAG,"获取详情:json="+ Gson().toJson(taskDetailToBean))
        HTTPProxy.instance!!.execute( this,taskDetailToBean, Constant.REQUEST_TASK_DETAIL);
    }

如注释所言,只要在订阅时使用

`as`(AutoDispose.autoDisposable(AndroidLifecycleScopeProvider.from(lifecycleOwner)))

就可以绑定lifecycleOwner,从而跟随生命周期自动取消订阅
这样,只要在网略请求回调时(这里给出栗子):

 //网略请求回调
    override fun <T> onResponse(observable: Observable<T>, typeLogin: Int) {
        try {
            //业务请求类型
            when (typeLogin) {
                //获取任务请求
                Constant.REQUEST_TASK -> taskLiveData.postValue(observable as (Observable<TaskBean>))
                //获取任务详情请求
                Constant.REQUEST_TASK_DETAIL ->taskDetailLiveData.postValue(observable as (Observable<TaskDetailBean>))
            }
        } catch (e :Exception){
            e.printStackTrace();
        }
    }

调用taskLiveData.postValue(observable as (Observable<TaskBean>))更新数据即可,taskLiveData只会在Activity处于活动状态下处理(常见网略请求回调后更新UI,此时如果UI所在的Activity已消亡就会引发错误),在liveData里面处理网略请求回调的数据就不会造成资源浪费,然后又在Activity消亡时自动取消订阅,就不会引发内存泄漏

  • View中使用
val testModel by lazy { ViewModelProviders.of(this).get(TestNodel::class.java) }//绑定lifeCycleOwner,获取ViewModel
...
testModel.initData(this)
testModel.requestTaskDetail(taskDetailToBean)

这里省略去部分代码,总之,在网略请求之后,会调用自定义的一个接口,回调上面给出的onResponse方法,然后更新liveData的数据。网略请求使用的是一个rxjava+retrofit+okhttp自己封装的工具,代码就不给出了。

  • 测试
    网上有段很内存泄漏的代码,很好测试内存泄漏,这里给出来以供测试时复制粘贴
Observable.interval(1, TimeUnit.SECONDS)
.subscribeOn(Schedulers.io())
.observeOn(AndroidSchedulers.mainThread())
.subscribe(object : io.reactivex.Observer<Long>{
    override fun onComplete() {
        LogUit.printI(RegisterOrLoginModel.TAG,"onComplete")
    }

    override fun onSubscribe(d: Disposable) {
        LogUit.printI(RegisterOrLoginModel.TAG,"onSubscribe:d=$d")
    }

    override fun onNext(t: Long) {
        LogUit.printI(RegisterOrLoginModel.TAG,"onNext:t=$t,")
    }

    override fun onError(e: Throwable) {
        LogUit.printI(RegisterOrLoginModel.TAG,"onError=${e.message}")
    }

})

测试的方法是这样的:定义两个Activity:A和B,从A跳转到B,B持有栗子中的TestViewModel,然后在taskLiveData.observe(```)方法的处理中添加这段代码,每个一秒就会打印log一次,然后按返回键回到A,停止Log打印,如果不加上

`as`(AutoDispose.autoDisposable(AndroidLifecycleScopeProvider.from(lifecycleOwner)))

则会一直打印log,发生内存泄漏。

3.注意

AutoDispose的版本已经更新到1.3.0了,如果使用超过1.0.0的版本(目前只测试到这个版本),绑定lifecycleOwner时就会要求使用androidx里面的lifecycleOwner,这将与LiveData的1.1.0版本冲突(笔者只试了下这个版本),因为LiveData要求的lifecycleOwner是android.arch.lifecycle.LifecycleOwner的,笔者尚未找到可以同时满足这两个条件的Activity(其实是没去找哈哈)
另外,本文代码使用kotlin写,还在使用java的要去更新一下技能了,使用kotlin不用担心以前的java代码不能使用,因为从java转换成kotlin非常容易!!!
原创文章

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,968评论 6 482
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,601评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 153,220评论 0 344
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,416评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,425评论 5 374
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,144评论 1 285
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,432评论 3 401
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,088评论 0 261
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,586评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,028评论 2 325
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,137评论 1 334
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,783评论 4 324
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,343评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,333评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,559评论 1 262
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,595评论 2 355
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,901评论 2 345