Java-RxJava2笔记

参考:
http://blog.csdn.net/maplejaw_/article/details/52442065
http://www.jianshu.com/nb/5864063

去年RxJava2.x发布了,与RxJava1.x相比,使用上有不少改动(只是API函数名改了,而使用流程思维不变),故在此记录笔记存档!

一.基本用法

1.创建Observable(被观察者/发布者/发射者)

(1)create()
Observable observable = Observable.create(new ObservableOnSubscribe<String>() {
    @Override
    public void subscribe(@NonNull ObservableEmitter<String> observableEmitter) throws Exception {
        observableEmitter.onNext("发布数据1");
        observableEmitter.onNext("发布数据2");
        observableEmitter.onError(new Throwable("发生错误"));          
        observableEmitter.onComplete(); //完成
    }
});

(2)just
Observable observable = Observable.just("发布数据1", "发布数据2");

(3)fromIterable, fromArray
ArrayList<String> list = new ArrayList<>();
list.add("发布数据1");
list.add("发布数据2");
Observable observable = Observable.fromIterable(list);

(4)range,第一个参数为起始值,第二个为发送的个数,如果为0则不发送,负数则抛异常
Observable observable = Observable.range(10, 5)

(5)defer,延期,有观察者订阅时才创建Observable
Observable observable = Observable.defer(new Callable<ObservableSource>() {
    @Override
    public ObservableSource call() throws Exception {
        return Observable.just("发布数据1","发布数据2");
    }
});

(6)interval,定时周期发布数据
 Observable observable = Observable.interval(500, TimeUnit.MILLISECONDS); //周期500ms

(7)timer,延迟发布数据
Observable observable = Observable.timer(300, TimeUnit.MILLISECONDS); //延迟300ms

(8)repeat,重复发布数据
Observable observable = Observable.just("发布数据1").repeat(3); //重复发布3次

2.创建Observer(观察者/订阅者/接收者)

(1).Observer完整形式
Observer observer = new Observer() {
    @Override
    public void onSubscribe(@NonNull Disposable disposable) {
        //Disposable 相当于RxJava1.x中的Subscription,用于解除订阅
    }

    @Override
    public void onNext(@NonNull Object o) {
        //接收数据
    }

    @Override
    public void onError(@NonNull Throwable throwable) {
        //接收错误
    }

    @Override
    public void onComplete() {
        //通知完成
    }
};

(2).Observer简写形式
Consumer onNext = new Consumer() {//接收数据
    @Override
    public void accept(Object o) throws Exception {
    }
};

Consumer<Throwable> onError = new Consumer<Throwable>() {//接收错误
    @Override
    public void accept(Throwable throwable) throws Exception {
    }
};

Action onComplete = new Action() {//通知完成
    @Override
    public void run() throws Exception {
    }
};

Consumer<Disposable> onSubscribe = new Consumer<Disposable>() {
    @Override
    public void accept(Disposable disposable) throws Exception {
    }
};

3.Observer订阅Observable

(1).Observer完整订阅
observable.subscribe(observer); //订阅

(2).Observer简写订阅
observable.subscribe(onNext);
observable.subscribe(onNext, onError);
observable.subscribe(onNext, onError, onComplete);
observable.subscribe(onNext, onError, onComplete, onSubscribe);

二.线程调度

调度器类型
Schedulers.computation( )  用于计算任务,如事件循环或和回调处理,不要用于IO操作(IO操作请使用Schedulers.io());默认线程数等于处理器的数量
Schedulers.from(executor)  使用指定的Executor作为调度器
Schedulers.io( )           用于IO密集型任务,如异步阻塞IO操作,这个调度器的线程池会根据需要增长;
                           对于普通的计算任务,请使用Schedulers.computation();
                           Schedulers.io( )默认是一个CachedThreadScheduler,很像一个有线程缓存的新线程调度器
Schedulers.newThread( )    为每个任务创建一个新线程
Schedulers.trampoline( )   当其它排队的任务完成后,在当前线程排队开始执行
AndroidSchedulers.mainThread()  此调度器为RxAndroid特有,顾名思义,运行在Android UI线程上

Observable.just("耗时操作...")
    .subscribeOn(Schedulers.io())//io线程-发布者
    .observeOn(AndroidSchedulers.mainThread())//主线程-接收者
    .subscribe(new Consumer<String>() {
        @Override
        public void accept(String s) throws Exception {                        
        }
    });

三.常用操作符

1.map-数据类型转换

Observable.just("123")
    .map(new Function<String, Integer>() {
        @Override
        public Integer apply(@NonNull String s) throws Exception {
            return Integer.parseInt(s);
        }
    })
    .subscribe(new Consumer<Integer>() {
        @Override
        public void accept(Integer integer) throws Exception {
            System.out.println(integer);
        }
    });

2.flatMap-数据集合扁平化(遍历循环每一个元素)

List<String> list = new ArrayList<>();
list.add("1");
list.add("2");
list.add("3");
List<List<String>> listSSS = new ArrayList<>();//二维数组集合
listSSS.add(list);
Observable.fromIterable(listSSS)
.flatMap(new Function<List<String>, ObservableSource<String>>() {
    @Override
    public ObservableSource<String> apply(@NonNull List<String> list) throws Exception {
        return Observable.fromIterable(list);
    }
})
.subscribe(new Consumer<String>() {
    @Override
    public void accept(String s) throws Exception {
        System.out.println(s);
    }
});

3.buffer-缓存满后,以list集合发送数据

List<String> list = new ArrayList<>();
list.add("1");
list.add("2");
list.add("3");
Observable.fromIterable(list)
    .buffer(list.size())  //缓存一起发送
    .subscribe(new Consumer<List<String>>() {
        @Override
        public void accept(List<String> list) throws Exception {
            System.out.println(list.size());
        }
    });

4.take(n)-发送前n项数据

Observable.just(1, 2, 1, 1, 2, 3)
    .take(3) //发送前3项数据
    .subscribe(new Consumer<Integer>() {
        @Override
        public void accept(Integer integer) throws Exception {
            System.out.println(integer);
        }
    });

5.distinct-去除重复项

Observable.just(1, 2, 1, 1, 2, 3)
    .distinct() //去重
    .subscribe(new Consumer<Integer>() {
        @Override
        public void accept(Integer integer) throws Exception {
            System.out.println(integer);
        }
    });

6.filter-过滤

Observable.just(1, 2, 3, 4, 5)
    .filter(new Predicate<Integer>() {
        @Override
        public boolean test(@NonNull Integer integer) throws Exception {
            return integer > 3; //过滤大于3
        }
    })
    .subscribe(new Consumer<Integer>() {
        @Override
        public void accept(Integer integer) throws Exception {
            System.out.println(integer);
        }
    });

五.Flowable-背压

Flowable是RxJava2.x中新增的类,专门用于应对背压Backpressure问题
背压: 即生产者的速度大于消费者的速度带来的问题,比如在Android中常见的点击事件,点击过快则会造成点击两次的效果!
Flowable.create(new FlowableOnSubscribe<Integer>() {
        @Override
        public void subscribe(FlowableEmitter<Integer> e) throws Exception {
            for (int i = 0; i < 10000; i++)
                e.onNext(i);
            e.onComplete();
        }
}, BackpressureStrategy.ERROR) //指定背压处理策略,抛出异常错
    .subscribeOn(Schedulers.computation())
    .observeOn(Schedulers.newThread())
    .subscribe(new Consumer<Integer>() {
        @Override
        public void accept(Integer integer) throws Exception {
            System.out.println(integer);
            Thread.sleep(1000);
        }
    }, new Consumer<Throwable>() {
        @Override
        public void accept(Throwable throwable) throws Exception {
            System.out.println(throwable);
        }
    });

// 如Rxjava1.x一样简写
Flowable.range(1,10000)
.onBackpressureDrop() // 背压
.subscribe(new Consumer<Integer>() {
    @Override
    public void accept(Integer integer) throws Exception {
        System.out.println(integer);
    }
});

四.Subject

Subject extends Observable implements Observe
作用:
    可充当Observable
    可充当Observer
    是Observable和Observer之间的桥梁        
Subject有四个实现类: AsyncSubject, BehaviorSubject, PublishSubject, ReplaySubject
注意:
    从多个线程中调用onNext(on系列方法),需要使用串行化Serialized,才能顺序调用!
    SerializedSubject<String, Integer> ser = new SerializedSubject(publishSubject);

Processor和Subject的作用相同,其中Processor是RxJava2.x新增的,继承自Flowable,所以支持背压控制
//Processor
AsyncProcessor<String> processor = AsyncProcessor.create();
processor.subscribe(o -> Log.d("JG",o)); //three
processor.onNext("one");
processor.onNext("two");
processor.onNext("three");
processor.onComplete();

1.AsyncSubject只接收onCompleted()被调用前的最后一个数据

AsyncSubject<String> asyncSubject = AsyncSubject.create();
asyncSubject.onNext("asyncSubject1");
asyncSubject.onNext("asyncSubject2");
asyncSubject.onNext("asyncSubject3");
asyncSubject.onComplete();
asyncSubject.subscribe(new Consumer<String>() {
    @Override
    public void accept(String s) throws Exception {
        System.out.println(s);//只接收到asyncSubject3
    }
});    

2.BehaviorSubject接收被订阅前的最后一个数据,还接收订阅后的数据

BehaviorSubject<String> behaviorSubject = BehaviorSubject.create();
behaviorSubject.onNext("behaviorSubject1");
behaviorSubject.onNext("behaviorSubject2");
behaviorSubject.subscribe(new Consumer<String>() {
    @Override
    public void accept(String s) throws Exception {
        System.out.println(s); //接收到behaviorSubject2, behaviorSubject3, behaviorSubject4
    }
});
behaviorSubject.onNext("behaviorSubject3");
behaviorSubject.onNext("behaviorSubject4");

3.PublishSubject只接收被订阅后的数据

PublishSubject<String> publishSubject = PublishSubject.create();
publishSubject.onNext("publishSubject1");
publishSubject.onNext("publishSubject2");
publishSubject.subscribe(new Consumer<String>() {
    @Override
    public void accept(String s) throws Exception {
        System.out.println(s);只接收到behaviorSubject3, behaviorSubject4
    }
});
publishSubject.onNext("publishSubject3");
publishSubject.onNext("publishSubject4");

4.ReplaySubject接收所有数据,无论何时订阅! 但缓存到一定大小时或一段时间后会丢弃旧的数据!

ReplaySubject<String> replaySubject = ReplaySubject.create(); //默认初始缓存容量大小为16
//replaySubject = ReplaySubject.create(100);//指定初始缓存容量大小为100
//replaySubject = ReplaySubject.createWithSize(2);//只缓存订阅前最后2条数据
//replaySubject = ReplaySubject.createWithTime(1,TimeUnit.SECONDS,Schedulers.computation());//只缓存被订阅前1秒内的数据
replaySubject.onNext("replaySubject:pre1");
replaySubject.onNext("replaySubject:pre2");
replaySubject.onNext("replaySubject:pre3");
replaySubject.subscribe(new Consumer<String>() {
    @Override
    public void accept(String s) throws Exception {
        System.out.println(s);
    }
});
replaySubject.onNext("replaySubject:after1");
replaySubject.onNext("replaySubject:after2");

5.Subject作为桥梁,使用示例

//1.Subject作为桥梁
Subject<String> subject = BehaviorSubject.create();

//2.订阅
subject.subscribe(new Consumer<String>() {
    @Override
    public void accept(String s) throws Exception {
        System.out.println(s);
    }
});

//3.发布数据
Observable.create(new ObservableOnSubscribe<String>() {
    @Override
    public void subscribe(@NonNull ObservableEmitter<String> observableEmitter) throws Exception {
        observableEmitter.onNext("as Bridge");
    }
}).subscribe(subject);

简书: http://www.jianshu.com/p/724c937e3d0c
CSDN博客: http://blog.csdn.net/qq_32115439/article/details/78090944
GitHub博客: http://lioil.win/2017/09/25/JavaSE-RxJava.html
Coding博客: http://c.lioil.win/2017/09/25/JavaSE-RxJava.html

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

推荐阅读更多精彩内容