手写 RxJava ---- map 操作符

一、实现的思路

RxJava 提供了很多的变换操作符,将上游的数据转换成另一种数据,可以在传送数据流的过程中构造一个新的类,这个类即持有上游的 Observable 也持有下游的 Observer 和 变换的能力来实现

二、具体代码

1.创建 map 操作符

map 操作符本身持有 ObservableOnSubscribe 对象,这里将 ObservableOnSubscribe 交给一个 ObservableMap 让它持有了上游的能力。同时也持有了变换的能力

  /**
     * 自定义 map 操作符
     * map 操作符会拿到 SelfObservableMap 所持有的 Observable 对象
     * 交给一个 ObservableMap 对象 然后替换一个新的(持有 ObservableMap 这个 ObservableOnSubscribe 的引用,已经一个转换的函数)
     * 即拥有控制上一层的能力 也拥有控制下一层的能力
     * <p>
     * 变换操作符只考虑上一层的类型 变换成新的类型后 给到下一层
     */
    public <R> SelfObserverable<R> map(CFunction<? super T, R> function) {
        ObservableMap observableMap = new ObservableMap(observable, function);
        // 这里会把 source 替换成 ObservableMap
        // 在 subscribe 时实际调用的 observable map 所持有的 source 的 subscribe 方案
        return new SelfObserverable<>(observableMap);
    }

2.ObservableMap 同时提供了 subscribe 函数,让它持有了下游的能力

/**
 * 为 Map 专门定义的 ObservableOnSubscribe
 * T 接收的类型 R 返回的类型
 *
 */
public class ObservableMap<T,R> implements ObservableOnSubscribe<R> {
    // 这里持有的是第一次通过 create 函数创建持有的 ObservableOnSubscribe 对象
    private ObservableOnSubscribe observable;
    private CFunction<? super T, ? extends R> function;

    public ObservableMap(ObservableOnSubscribe source, CFunction<? super T, ? extends R> function) {
        this.observable = source;
        this.function = function;
    }

    /**
     * observableEmitter 是外层通过 subscribe 函数传递进来的 Observer 类
     * @param observer
     */
    @Override
    public void subscribe(Observer<? super R> observer) {
        // SelfObserverable 调用 subscribe 函数时 实际上调用的是
        MapObserver mapObserver = new MapObserver(observer,  function);
        // observable 是通过 create 或者 just 传递进来的 observable,现在交给了它一个 MapObserver
        /**
         *     observable ----> create 的 new ObservableOnSubscribe
         *     // 使用 Map 操作符
         *     SelfObserverable.create(new ObservableOnSubscribe<Integer>() {
         *      @Override
         *      public void subscribe(Observer<? super Integer> observableEmitter) {
         *            observableEmitter.onNext(1);
         *            observableEmitter.onComplete();
         *         }
         *     })
         *
         *    这个 observable.subscribe 调用的就是 ObservableOnSubscribe 的 subscribe 函数的方法。
         *    会触发 MapObserver 类的 onNext 和 onComplete
         */
        observable.subscribe(mapObserver);
    }

3.在 subscribe 时又对 observer 做了进一步的包装,让被观察者 ObservableOnSubscribe 持有了 MapObserver 这个新的观察者类。在调用 onNext 时首先调用 CFunction 函数进行一次数据转换

 /**
     * 对 Observer 的包装
     *
     * @param <T>
     */
    class MapObserver<T> implements Observer<T> {
        Observer<? super R> observableEmitter;
        ObservableOnSubscribe source;
        CFunction<? super T, ? extends R> function;

        /**
         *
         * @param observableEmitter  通过 subscribe 传递进来的观察者
         * @param function function 是负责转换的函数
         */
        public MapObserver(Observer<? super R> observableEmitter,
                           CFunction<? super T, ? extends R> function) {
            this.observableEmitter = observableEmitter;
            this.function = function;
        }

        @Override
        public void onSubscirbe() {
            observableEmitter.onSubscirbe();
        }

        @Override
        public void onNext(T value) {
            // 将转换后的值交给 onNext
            R next = function.apply(value);
            observableEmitter.onNext(next);
        }

        @Override
        public void onComplete() {
            observableEmitter.onComplete();
        }

        @Override
        public void onError(Throwable throwable) {
            observableEmitter.onError(throwable);
        }
    }

四、流程示例

RxJava-2.jpg

五、使用示例

  // 使用 Map 操作符
        SelfObserverable.create(new ObservableOnSubscribe<Integer>() {
            @Override
            public void subscribe(Observer<? super Integer> observableEmitter) {
                observableEmitter.onNext(1);
                observableEmitter.onComplete();
            }
        }).map(new CFunction<Integer, String>() {

            @Override
            public String apply(Integer integer) {
                return "我被转换啦";
            }
        }).subscribe(new Observer<String>() {
            @Override
            public void onSubscirbe() {
                Log.d("TAG", "on subscribe with map");
            }

            @Override
            public void onNext(String value) {
                Log.d("TAG", "我被转换啦");
            }

            @Override
            public void onComplete() {

            }

            @Override
            public void onError(Throwable throwable) {

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

推荐阅读更多精彩内容