RxJava 2.0 笔记

  • 简单的介绍
  • 简单的用法
  • 各操作符
  • 实际的项目运用

介绍

很多博客都写的很清楚了,先推荐一些入门挺好的博客:
关于 RxJava 最友好的文章—— RxJava 2.0 全新来袭
给 Android 开发者的 RxJava 详解

然后说说这个东西的核心:观察者模式

图片源自[关于RxJava最友好的文章](https://gold.xitu.io/post/580103f20e3dd90057fc3e6d)

这里的 Observe 会被 Observable 的事件主动的触发。
也就是 Observe 的方法不会被显性的调用,而是等待 Observable 的呼叫。

其实关于这个相信大家都不是第一次接触,比如 ListView 、 Button 的OnClick 等事件,就是利用的观察者模式。这时候 ListView 就是 Observable ,OnClick 事件就是 Ovserve

当然 RxJava 最厉害的还是它的异步操作
相对于其他的异步的方法 AsyncTask 、Handler 等,它的流式API调用风格和各种操作符可以让程序越来越复杂的情况下依然保持代码的清晰,
注意它并不能让代码变得简单,而是清晰,也就是易读

RxJava 1 和 RxJava 2 的区别最大在于 背压 的实现方法变了

关于 RxJava 最友好的文章——背压(Backpressure)

简单介绍这个概念:
在原来框架中,被观察者 Observable 会主动向 观察者 Observer 发送事件。
而利用背压这个策略的时候,Observer 会向 Observable 发送请求,Observable 收到请求后才会发送事件。
原因是如果 Observer 处理事件的速度远远慢于 Observable 发送的速度,会使得内存泄露。

基础的方法

创建 Observable 和 Observer
不支持背压

最基础的调用:

Observable mObservable=Observable.create(new ObservableOnSubscribe<Integer>() {
            @Override
            public void subscribe(ObservableEmitter<Integer> e) throws Exception {
                e.onNext(1);
                e.onNext(2);
                e.onComplete();
            }
        });

Observer mObserver=new Observer<Integer>() {

            @Override
            public void onSubscribe(Disposable d) {
              //订阅之后、发送数据之前被调用
             //可以使用d.dispose()方法来取消订阅
            }

            @Override
            public void onNext(Integer value) {

            }

            @Override
            public void onError(Throwable e) {

            }

            @Override
            public void onComplete() {

            }
        };

mObservable.subscribe(mObserver);//这句如果漏掉了就不能让这两个对象产生联系

创建 Flowable 和 Subscriber
支持背压

Flowable.create(new FlowableOnSubscribe<String>() {
    @Override
    public void subscribe(FlowableEmitter<String> e) throws Exception {
        e.onNext("Hello,I am China!");
    }
}, BackpressureStrategy.BUFFER).subscribe(subscriber); 
//需要指定背压策略,BUFFER

Subscriber<String> subscriber = new Subscriber<String>() {
    @Override
    public void onSubscribe(Subscription s) {
        //初始化操作,调用 request() 方法表示初始化工作已经完成,立即触发 onNext () 方法
        //在 onComplete() 方法完成,才会再执行 request() -  后边的代码

        Log.w("TAG","onsubscribe start");  
        s.request(1);//请求数据
        Log.w("TAG","onsubscribe end");
    }

    @Override
    public void onNext(String value) {
        Log.e("onNext", value);
        sub.request(1);//继续请求数据
    }

    @Override
    public void onError(Throwable t) {
        Log.e("onError", t.getMessage());
    }

    @Override
    public void onComplete() {
        Log.e("onComplete", "complete");
    }
};

上面的代码输出语句顺序为:

onsubscribe start
onNext
onComplete
onsubscribe end

FLowable 和 Observable 还有很多别的创建方式:

//第一个参数为起始值,第二个为发送的个数,如果为0则不发送,负数则抛异常。
//即调用20次 onNext() 方法,依次传入数字 1 - 20
Flowable f = Flowable.range(0,10);

// 按固定时间间隔发射整数序列,即按照固定2秒一次调用onNext()方法。
//TimeUnit 是 java 工具类,可以换单位 TimeUnit.HOURS
Observable o = Observable.interval(2, TimeUnit.SECONDS)

Observable<String> o = Observable.just("Hello");

不一一列出

各类型操作符与实际的项目运用

类型转换

  • flatMap() //与 map 类似,不同于用于将一个对象加工成另一个对象的数组
  • map() //在事件传递过程中,转换传递的对象的类型
  • filter() // 根据重写的 test() 方法中的判断语句,返回 true 则表示数据满足条件,返回 false 则表示数据需要被过滤。

线程变换

  • subscribeOn()//它指示 Observable 在一个指定的调度器上创建(只作用于被观察者创建阶段)。只能指定一次,如果指定多次则以第一次为准

  • observeOn()//指定在事件传递(加工变换)和最终被处理(观察者)的发生在哪一个调度器。可指定多次,每次指定完都在下一步生效。

下面是一个实际的操作,流式 api 调用风格:
由 String 型的 url 通过异步网络请求后,将得到的 JSON 转换成想要的数据
上面的操作符都有详细的注释

String News_URL = "http://xxxx";

//这一句创建一个 observable 对象,并用将 News_URL 传递下去
Observable.fromArray(News_URL)

//指定在新线程中创建被观察者 observable
.subscribeOn(Schedulers.newThread())

//map 将 fromArray() 传入的 News_URL (String) 变成 JSONObject 对象
.map(new Function<String, JSONObject>() {
    @Override
    public JSONObject apply(String s) throws Exception {
        //参数 s 就是上面传进来的 News_URL
        URL new_url=new URL(s);
        //调用自己写的方法,一些网络请求,不是重点
        JSONObject jsonObject= getData.URL2JSONObject(new_url);
        //返回 JSONObject 类型的对象,完成了转换
        return jsonObject;
    }
})

//map 返回的 JSONObject 对应这里的第一个参数
.flatMap(new Function<JSONObject, ObservableSource<JSONObject>>() {
    @Override
    public ObservableSource<JSONObject> apply(JSONObject jsonObject) throws Exception {
      
        //jsonObject 就是 map 返回的对象
        JSONArray stories = jsonObject.getJSONArray("stories");
        JSONObject[] jsonObjectsList = new JSONObject[stories.length()];
 
        //这一段不是重点,
        for (int i = 0;i < stories.length();i++){
            jsonObjectsList[i]= stories.getJSONObject(i);
        }

        //flatMap 的重点在于这里
        //看上面的 Function<JSONObject, ObservableSource<JSONObject> 可以知道
        //flatMap 返回的对象是由第二个参数指定,即 ObservableSource<JSONObject>
        //但这里实际 return 了一个 JSONObject [] ,一个数组
        //因为 flatMap 会把这个数组里的对象逐个传递到下面的 onNext 中
        return Observable.fromArray(jsonObjectsList);

    }
})

//切换线程,将这句代码以下的操作放到主线程中
.observeOn(AndroidSchedulers.mainThread())

.subscribe(new Observer<JSONObject>() {
    @Override
    public void onSubscribe(Disposable d) {

    }
    @Override
    public void onNext(JSONObject jO) {
        try {
            //这一段代码会被多次执行
            //执行次数由上面的 flatMap 里的 jsonObjectsList 的长度决定
            News news=new News();
            String url = null;
            url = jO.getJSONArray("images").getString(0).toString();
            news.setImages(url);
            news.setTitle(jO.getString("title"));
            news.setId(jO.getString("id"));
            newss.add(news);

        } catch (JSONException e) {
            e.printStackTrace();
        }
    }
    @Override
    public void onError(Throwable e) {
    }
    @Override
    public void onComplete() {
        ListView newslistView=(ListView) view.findViewById(R.id.show_news);
        newAdapter = new NewsAdapter(newss,mcontext,newslistView);
        newslistView.setAdapter(newAdapter);

    }
});

关于线程的切换:
下面这段代码来自关于 RxJava 最友好的文章

          //new Observable.just()执行在新线程
  Observable.just(getFilePath())
           //指定在新线程中创建被观察者
          .subscribeOn(Schedulers.newThread())
          //将接下来执行的线程环境指定为io线程
          .observeOn(Schedulers.io())
            //map就处在io线程
          .map(mMapOperater)
            //将后面执行的线程环境切换为主线程,
            //但是这一句依然执行在io线程
          .observeOn(AndroidSchedulers.mainThread())
          //指定线程无效,但这句代码本身执行在主线程
          .subscribeOn(Schedulers.io())
          //执行在主线程
          .subscribe(mSubscriber);

感谢 拉丁吴 的博客,他还写有另一篇关于 RxJava 的进阶内容:
关于RxJava最友好的文章(进阶)

这里都是简单的笔记,后续如果深入源码会继续更新。

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

推荐阅读更多精彩内容