美文网首页RxjavaRxJavaRxJava系列专题(Android方向)
RxJava使用takeUntil操作实现生命周期的绑定

RxJava使用takeUntil操作实现生命周期的绑定

作者: middle2021 | 来源:发表于2016-07-03 00:50 被阅读2923次

相信使用过RxJavaRxAndroid的人都会被其简便的流式操作所吸引,结合java8的lambda表达式,使代码看起来更简洁,操作逻辑更清晰,数据处理更便捷,告别了臃肿的代码。但要优雅的使用它还需要进行一些处理,避免操作不当导致的内存泄漏风险。


管理订阅的几种方式
  • 使用CompositeSubscription统一管理生命周期(后面具体讲解实现方式)
  • 使用RxLifecycle来管理订阅,实现和Android组件的生命周期绑定。
  • 使用takeUntil操作符结合BehaviorSubject实现生命周期绑定。

具体介绍

一、使用CompositeSubscription统一管理生命周期
public class BaseActivity extends AppCompatActivity {    
    protected CompositeSubscription subscription = new   CompositeSubscription();    
    protected void addSub(Subscription sub) {        
        if (sub != null && !sub.isUnsubscribed()) {                   
          subscription.add(sub);       
     }   
 }    
@Override   
protected void onDestroy() {     
     if(subscription.hasSubscriptions()){          
          subscription.unsubscribe();        
    }        
    super.onDestroy();   
 }
}

在子类统一调用addSub方法将生成的订阅关系添加进subscription管理类中,在onDestroy中统一取消订阅关系。

二、使用RxLifecycle来管理订阅(需要引入新的类库)

这里不进行详细描述,具体操作参照Rxlifecycle使用详解

三、使用takeUntil操作符结合BehaviorSubject实现生命周期绑定

具体代码实现:

//这里提供的是kotlin的实现
class MainActivity : AppCompatActivity() {    
  var subject: BehaviorSubject<Event> = BehaviorSubject.create()    
  val TAG = MainActivity::class.java!!.simpleName    
  override fun onCreate(@Nullable savedInstanceState: Bundle?) {   
              super.onCreate(savedInstanceState)        
              setContentView(R.layout.activity_main)       
     var sub = Observable.interval(1, TimeUnit.SECONDS)
    .compose(this.bindLife<Long>())
    .subscribe({ a -> Log.i(TAG, "call: " + a!!) },
    { t -> Log.e(TAG, "call: " + t.message!!) }                       
   , { Log.i(TAG, "complete: 完成") })
}    

  fun <T> bindLife(): Observable.Transformer<T, T> { 
    return Observable.Transformer<T, T> {           
     observable -> observable.takeUntil(subject.skipWhile(
    {event ->event != Event.PAUSE && event != Event.DESTROY
    && event != Event.DETACH }))}    
  }   
 
  override fun onDestroy() { 
    subject.onNext(Event.DESTROY)
    super.onDestroy()    
  }
}

enum class Event {    
// Activity life Events    
  CREATE,   
  START,  
  RESUME,  
  PAUSE,  
  STOP,   
  DESTROY,    
// Fragment life  Events  
  ATTACH, 
  CREATE_VIEW,  
  DESTROY_VIEW,  
  DETACH
}

java版本(非lambda)

/**
 * ClassName: BaseActivity<p>
 * Author: Alpha<p>
 * Fuction: <p>
 * CreateDate: 2016/7/3 0:17<p>
 */
public class BaseActivity extends AppCompatActivity {

    protected BehaviorSubject<Event> subject = BehaviorSubject.create();

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        Observable.interval(3, TimeUnit.SECONDS)
                .compose(this.<Long>bindLife())
                .observeOn(AndroidSchedulers.mainThread())
                .subscribe(new Subscriber<Long>() {
                    @Override
                    public void onCompleted() {
                        Log.i("rxjava", "onCompleted");

                        Toast.makeText(BaseActivity.this, "onCompleted", Toast.LENGTH_SHORT).show();

                    }

                    @Override
                    public void onError(Throwable e) {

                    }

                    @Override
                    public void onNext(final Long aLong) {
                        Log.i("rxjava", "onNext " + aLong);

                        Toast.makeText(BaseActivity.this, "aLong:" + aLong, Toast.LENGTH_SHORT).show();
                    }
                });
    }

    protected <T> Observable.Transformer<T, T> bindLife() {
        return new Observable.Transformer<T, T>() {
            @Override
            public Observable<T> call(Observable<T> observable) {
                return observable.takeUntil(subject.skipWhile(new Func1<Event, Boolean>() {
                    @Override
                    public Boolean call(Event event) {
                        return event != Event.DESTROY && event != Event.DETACH;
                    }
                }));
            }
        };
    }

    @Override
    protected void onDestroy() {
        subject.onNext(Event.DESTROY);
        super.onDestroy();
    }
}

未处理之前.gif 处理之后.gif

鸣谢:#Kotlin# 小心 Rx 的生命周期

感谢大家的支持,如有不足之处希望大家提出来,一起交流进步。

相关文章

  • RxJava使用takeUntil操作实现生命周期的绑定

    相信使用过RxJava、RxAndroid的人都会被其简便的流式操作所吸引,结合java8的lambda表达式,使...

  • RxJava相关

    Room ? RxJava使用RxJava操作数据库 RxPermissionsRxJava实现的Android运...

  • ReactiveCocoa使用心得

    单向绑定 在cell中对控件进行绑定,需要添加takeUntil 双向绑定 RACCommand 订阅: RACS...

  • RxJava 线程控制

    引言 该篇文章主要是关于RxJava的线程使用的代码demo讲解。 具体使用 在RxJava中我们实现操作符sub...

  • 使用RxJava实现定时器功能

    使用RxJava实现定时器功能可以通过两种方式来实现,具体实现如下: 一、使用timer 操作符 二、使用使用in...

  • takeUntil操作符探究

    之前看了下RxLifecycle的实现,短短几行代码,屌到不行。其中核心实现就是takeUntil操作符,然后我就...

  • RxJava 2.0----观察者模式之厨师与顾客

    一. 引言RxJava ,简单来说,一个实现异步操作的库。代替了我以前使用Thread,AsyncTask实现异步...

  • RxJava

    其它文章 RxJava操作符大全 1、RxJava之一——一次性学会使用RxJava RxJava简单的使用和使用...

  • Rxjava (上)

    概述: Rxjava作用:异步,rxjava是实现异步操作的库Rxjava好处:无论业务逻辑多么复杂,rxjava...

  • RxJava 源码浅析

    本文只分析 RxJava 的基本原理与流程,不深入探讨具体操作符的实现细节。 背景 为什么使用 RxJava? 解...

网友评论

  • 8f882715cd99:楼主能解释下为什么要先takeUntil再skipWhile吗?我的理解就是当视图destroy的时候,将f设置为true,然后只提供一个skipWhile返回这个f就行了,但实际上测试并未达到预期的效果,理解不能,求楼主科普下。
    middle2021: @xsjiang 你直接用标记, 还是要在视图销毁的生命周期方法发送事件的
    middle2021: @xsjiang 这个是因为skipWhile是过滤的,视图销毁会发送事件的,这时skipWhile匹配到了,返回true,就会将事件发送出去,而takeUntil会接受到事件,就会终止对应的外层的Obserable的事件,解除订阅关系
  • Euterpe:感觉 写给别人看 最好别用lambda先
    middle2021:@Euterpe 嗯,写的时候没有考虑到,谢谢提醒。

本文标题:RxJava使用takeUntil操作实现生命周期的绑定

本文链接:https://www.haomeiwen.com/subject/vdnqjttx.html