美文网首页RxJavaAndroid技术知识
RxJava框架的学习总结---基础用法01

RxJava框架的学习总结---基础用法01

作者: Elder | 来源:发表于2016-12-18 22:18 被阅读68次

    之前看过RxJava的一篇文章非常不错跳转博文通过这篇博文我相信大家都会对RxJava这个框架的使用有了一定的心得。这里将自己的学习的内容进行总结,方便后期查用。

    一、RxJava简介

    GitHub地址大家可以自行了解,这里我简单介绍一下Android Studio如何添加RxJava依赖:

    compile 'io.reactivex:rxjava:1.2.4'
    compile 'io.reactivex:rxandroid:1.2.1'
    

    这是截止目前最新的1.x版本了,在GitHub上可以看见RxJava已经有了2.0的更新版,由于是初学者并没有那么急于求成,还是从1.0的版本开始学习使用吧。这里你还可以看到RxAndroid的身影,简单介绍一下:

    RxAndroid是对RxJava在Android上的扩展,如果你是做安卓开发的话,各种主线程和子线程的操作肯定会让你觉得头疼,RxAndroid可以很容易地解决你的这种困扰。

    二、初步的使用

    RxJava 的异步实现,是通过一种扩展的观察者模式来实现的。观察者最简单的对象关系如下图,当一个被观察者发出事件时,观察者接收事件并调用回调方法响应此事件。


    观察模式关系图

    1、观察者(Observer)

    所以我们先定义一个观察者对象observer,代码如下:

    Observer<String> observer = new Observer<String>() {
                @Override
                public void onCompleted() {
    
                }
    
                @Override
                public void onError(Throwable e) {
    
                }
    
                @Override
                public void onNext(String s) {
    
                }
            };
    

    Observer是一个接口内部定义了三个方法如上,这里说明一下RxJava 不仅把每个事件单独处理,还会把它们看做一个队列。

    • onNext()当事件队列里有事件发出时即会调用此方法来处理
    • onError()当事件队列在处理过程中出异常时,onError()会被触发,同时队列自动终止,不允许再有事件发出。
    • onCompleted()当事件队列完结,并且不再有新事件发出时即会调用此方法。
    • 在一个正确运行的事件序列中, onCompleted()onError()有且只有一个,并且是事件序列中的最后一个。需要注意的是,onCompleted()onError()二者也是互斥的,即在队列中调用了其中一个,就不应该再调用另一个。

    除了Observer接口之外,RxJava还内置了一个实现了Observer接口的抽象类Subscriber,对Observer接口进行了一些扩展,但他们的基本使用方式是完全一样的。定义Subscriber对象的代码如下:

    subscriber = new Subscriber<String>() {
                @Override
                public void onStart() {
                    super.onStart();
                    // 这是一个扩展的方法,用于初始化一些数据,与定义Subscriber对象运行在同一个线程
                }
    
                @Override
                public void onCompleted() {
                    Log.e("xns", "onCompleted");
                }
    
                @Override
                public void onError(Throwable e) {
                    Log.e("xns", e.toString());
                }
    
                @Override
                public void onNext(String s) {
                    Log.e("xns", s);
                }
            };
    
     @Override
        protected void onStop() {
            if (subscriber.isUnsubscribed()) {// 释放资源的方法,防止内存泄漏
                subscriber.unsubscribe();
            }
            super.onStop();
        }
    
    • onStart()
      这是 Subscriber增加的方法。它会在 subscribe 刚开始,而事件还未发送之前被调用,可以用于做一些准备工作,例如数据的清零或重置。这是一个可选方法,默认情况下它的实现为空。需要注意的是,它总是在 subscribe 所发生的线程被调用,而不能指定线程。
    • unsubscribe()
      这是 Subscriber所实现的另一个接口 Subscription的方法,用于取消订阅。在这个方法被调用后,Subscriber将不再接收事件。一般在这个方法调用前,可以使用 isUnsubscribed()先判断一下状态。 unsubscribe()这个方法很重要,因为在 subscribe()之后, Observable会持有 Subscriber的引用,这个引用如果不能及时被释放,将有内存泄露的风险。所以最好保持一个原则:要在不再使用的时候尽快在合适的地方(例如 onPause()、onStop()等方法中)调用 unsubscribe()来解除引用关系,以避免内存泄露的发生。

    关于观察者创建和使用就介绍到这,我们再来看看被观察者的创建和使用方法。

    2、被观察者(Obserable)

    RxJava使用create()方法来创建一个Observable对象,并为它定义触发事件时的规则:

    Observable<String> observable = Observable.create(new Observable.OnSubscribe<String>() {
            @Override
            public void call(Subscriber<? super String> subscriber) {
                subscriber.onNext("Hello");
                subscriber.onNext("RxJava");
                subscriber.onCompleted();
            }
        });
    

    这里传入了一个OnSubscribe对象作为参数。它的作用相当于一个计划表,当Obserable被订阅的时候,OnSubscribe的call()方法就会被调用,就会依次调用订阅者的onNext()方法两次和onCompleted()方法。这样,就完成了被观察者调用观察者的回调方法来完成事件的传递,即观察者模式。

    • onCreate()是RxJava最基本的创建事件序列的方法。基于这个方法,RxJava还提供了一些快捷方法来创建。
    • just(T...)将传入的参数依次发送出来。
    Observable<String> observable2 = Observable.just("Hello","RxJava");
    
    • from(T[])/from(Iterable<? extends T>)将传入的数组或 Iterable拆分成具体对象后,依次发送出来。
    String[] strArr = {"Hello","RxJava"};
        Observable<String> observable3 = Observable.from(strArr);
    

    以上这三种方式都是等价的。

    3、订阅(Subscribe)

    创建了 Observable 和 Observer 之后,再用 subscribe() 方法将它们联结起来,整条链子就可以工作了。代码形式很简单:

    observable.subscribe(observer);
    // 或者:
    observable.subscribe(subscriber);
    

    Observable.subscribe(Subscriber) 的内部实现是这样的(仅核心代码):

    // 注意:这不是 subscribe() 的源码,而是将源码中与性能、兼容性、扩展性有关的代码剔除后的核心代码。
    // 如果需要看源码,可以去 RxJava 的 GitHub 仓库下载。
    public Subscription subscribe(Subscriber subscriber) {
        subscriber.onStart();
        onSubscribe.call(subscriber);
        return subscriber;
    }
    

    可以看到,subscriber() 做了3件事:

    • 调用 Subscriber.onStart() 。这个方法在前面已经介绍过,是一个可选的准备方法。
    • 调用 Observable 中的 OnSubscribe.call(Subscriber) 。在这里,事件发送的逻辑开始运行。从这也可以看出,在 RxJava 中, Observable 并不是在创建的时候就立即开始发送事件,而是在它被订阅的时候,即当 subscribe() 方法执行的时候。
    • 将传入的 Subscriber 作为 Subscription 返回。这是为了方便 unsubscribe()。(看源码你会发现Subscriber是Subscription的实现子类,Subscription接口只定义了两个方法void unsubscribe();boolean isUnsubscribed();
    被观察者与观察者方法的调用关系

    除了 subscribe(Observer) 和 subscribe(Subscriber) ,subscribe() 还支持不完整定义的回调,RxJava 会自动根据定义创建出 Subscriber 。形式如下:

    Action1<String> onNextAction = new Action1<String>() {
        // onNext()
        @Override
        public void call(String s) {
            Log.d(tag, s);
        }
    };
    Action1<Throwable> onErrorAction = new Action1<Throwable>() {
        // onError()
        @Override
        public void call(Throwable throwable) {
            // Error handling
        }
    };
    Action0 onCompletedAction = new Action0() {
        // onCompleted()
        @Override
        public void call() {
            Log.d(tag, "completed");
        }
    };
    
    // 自动创建 Subscriber ,并使用 onNextAction 来定义 onNext()
    observable.subscribe(onNextAction);
    // 自动创建 Subscriber ,并使用 onNextAction 和 onErrorAction 来定义 onNext() 和 onError()
    observable.subscribe(onNextAction, onErrorAction);
    // 自动创建 Subscriber ,并使用 onNextAction、 onErrorAction 和 onCompletedAction 来定义 onNext()、 onError() 和 onCompleted()
    observable.subscribe(onNextAction, onErrorAction, onCompletedAction);
    

    这里介绍一下Action0Action1。它们都是RxJava的内部接口,Action0接口只用一个call()方法,这个方法无参无返回值;由于 onCompleted() 方法也是无参无返回值的,因此 Action0 可以被当成一个包装对象,将 onCompleted() 的内容打包起来将自己作为一个参数传入 subscribe()以实现不完整定义的回调。这样其实也可以看做将 onCompleted() 方法作为参数传进了subscribe(),相当于其他某些语言中的『闭包』。 Action1接口有一个call(T param)方法,这个方法也无返回值可以看作是将 onNext(obj)onError(error) 打包起来传入 subscribe() 以实现不完整定义的回调。RxJava 还提供了多个 ActionX 形式的接口 (例如 Action2, Action3) 的,它们可以被用以包装不同的无返回值的方法。

    4、举例说明:

    a.打印字符串数组
    将字符串数组 names中的所有字符串依次打印出来:

    private void testRxjava2() {
            String[] names = {"xns","wxs","tyc"};
            Observable.from(names).subscribe(new Action1<String>() {
                @Override
                public void call(String s) {
                    Log.d("xns", s);
                }
            });
        }
    

    b. 由 id 取得图片并显示
    由指定的一个 drawable 文件 id drawableRes 取得图片,并显示在 ImageView 中,并在出现异常的时候打印 Toast 报错:

    private void testRxJava3() {
            final int drawableResId = 0;
            final ImageView imageView = new ImageView(this);
            Observable.create(new Observable.OnSubscribe<Drawable>() {
    
                @RequiresApi(api = Build.VERSION_CODES.LOLLIPOP)
                @Override
                public void call(Subscriber<? super Drawable> subscriber) {
                    Drawable drawable = getTheme().getDrawable(drawableResId);
                    subscriber.onNext(drawable);
                    subscriber.onCompleted();
                }
            }).subscribe(new Observer<Drawable>() {
                @Override
                public void onCompleted() {
    
                }
    
                @Override
                public void onError(Throwable e) {
    
                }
    
                @Override
                public void onNext(Drawable drawable) {
                    imageView.setImageDrawable(drawable);
                }
            });
        }
    
        private void testRxjava2() {
            String[] names = {"xns","wxs","tyc"};
            Observable.from(names).subscribe(new Action1<String>() {
                @Override
                public void call(String s) {
                    Log.d("xns", s);
                }
            });
        }
    

    以上就是RxJava的基本使用。然而在 RxJava 的默认规则中,事件的发出和消费都是在同一个线程的。也就是说,如果只用上面的方法,实现出来的只是一个同步的观察者模式。观察者模式本身的目的就是『后台处理,前台回调』的异步机制,因此异步对于 RxJava 是至关重要的。而要实现异步,则需要用到 RxJava 的另一个概念: Scheduler(调度器)。下一篇我将和大家继续学习RxJava接下来的内容。

    相关文章

      网友评论

        本文标题:RxJava框架的学习总结---基础用法01

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