新手读的懂的RxSwift源码解析(三)-- FlatMap

今天,笔者与大家分析一下,RxSwift中另一个特别重要,甚至是最重要之一的操作符:flatMap。

如果大家没有读过本系列的第一篇,建议先阅读一下再来看本篇内容。

在RxSwift进行开发的过程中,有一个非常常见的场景:点击某个按钮,然后发送请求,之后处理请求结果。代码大致如下:

myBtn
  .rx
  .tap
  .flatMap { _ in
    performRequest()
  }
  .map {
    processResponse($0)
  }
  .subScribe(
    onNext: { ... }
  )
  .disposed(by: bag)

在这个场景中,所要使用的最核心的一个操作符便是flatMap,它的作用是将myBtn产生的点击事件转换成另一个事件序列,并将该事件序列产生的事件向下传递。代码如下:

extension ObservableType {
    public func flatMap<Source: ObservableConvertibleType>(_ selector: @escaping (Element) throws -> Source)
        -> Observable<Source.Element> {
            return FlatMap(source: self.asObservable(), selector: selector)
    }
}

该方法接收一个(Element) throws -> Source的closure,Element就是自身的元素类型,而返回值类型Source是一个泛型参数,并且是遵循ObservableConvertibleType的。也就是说,这个closuer会把上游产生的事件转换成一个事件序列。方法实现中返回了一个FlatMap类型的对象,这个对象持有了self.asObservable(),以及传入的closure.
接下来,咱们看一下这个FlatMap的源码:

final private class FlatMap<SourceElement, SourceSequence: ObservableConvertibleType>: Producer<SourceSequence.Element> {
    typealias Selector = (SourceElement) throws -> SourceSequence

    private let source: Observable<SourceElement>
    
    private let selector: Selector

    init(source: Observable<SourceElement>, selector: @escaping Selector) {
        self.source = source
        self.selector = selector
    }
    
    override func run<Observer: ObserverType>(_ observer: Observer, cancel: Cancelable) -> (sink: Disposable, subscription: Disposable) where Observer.Element == SourceSequence.Element {
        let sink = FlatMapSink(selector: self.selector, observer: observer, cancel: cancel)
        let subscription = sink.run(self.source)
        return (sink: sink, subscription: subscription)
    }
}

FlatMap是一个Producer的子类,在它的run方法中通过self.selector和observer生成了一个FlatMapSink类的对象。这个selector就是在调用flatMap时传入的closure, observer其实就是在subscribe方法中创建的observer(参见系列第一篇)。然后调用了FlatMapSink的run方法。但是我们再FlatMapSink里并没有看到run方法,因为FlatMapSink是MergeSink的子类,它的很多逻辑都在它的父类MergeSink里面:

private class MergeSink<SourceElement, SourceSequence: ObservableConvertibleType, Observer: ObserverType>
    : Sink<Observer>
    , ObserverType where Observer.Element == SourceSequence.Element {
    typealias ResultType = Observer.Element
    typealias Element = SourceElement

    let lock = RecursiveLock()

    var subscribeNext: Bool {
        true
    }

    // state
    let group = CompositeDisposable()
    let sourceSubscription = SingleAssignmentDisposable()

    var activeCount = 0
    var stopped = false

    ......    
    
  func run(_ source: Observable<SourceElement>) -> Disposable {
        _ = self.group.insert(self.sourceSubscription)

        let subscription = source.subscribe(self)
        self.sourceSubscription.setDisposable(subscription)
        
        return self.group
    }
}

在源码的最后面,我们看到了run方法,只有四行,我们需要关注的是第二行:

let subscription = source.subscribe(self)

也就是说让self订阅了传入的Observable,而这个传入的Observable就是FlatMap.source,也就是上游序列。这样一来大致流程就清楚了:
通过Obsevable.flatMap,让FlatMap类持有了上游Obsevable,并通过FlatMapSink订阅了这个上游Obsevable。接下来上游Obsevable的所有事件都会被传递给FlatMapSink的on方法(原理参见系列第一篇)。
接下来的重点,我们需要看一下这个FlatMapSink的on方法做了什么,还是在父类MergeSink里面。

    func on(_ event: Event<SourceElement>) {
        switch event {
        case .next(let element):
            if let value = self.nextElementArrived(element: element) {
                self.subscribeInner(value.asObservable())
            }
        case .error(let error):
            self.lock.performLocked {
                self.forwardOn(.error(error))
                self.dispose()
            }
        case .completed:
            self.lock.performLocked {
                self.stopped = true
                self.sourceSubscription.dispose()
                self.checkCompleted()
            }
        }
    }

error事件会直接调用forwardOn,将事件传递给下游observer,completed事件则是调用checkCompleted方法,在checkCompleted方法中判断是否需要将completed事件传递下去。这是因为MergeSink可能会订阅多个事件序列,对这些事件进行一系列变换,直到所有的事件序列都complete之后才会传递,但这个不是今天的重点,所以只要知道个大概就可以了,具体的咱们在后续的文章中再介绍。
接下来看next事件,在next事件中会调用self.nextElementArrived(element: element):

    final private func nextElementArrived(element: SourceElement) -> SourceSequence? {
        self.lock.performLocked {
            if !self.subscribeNext {
                return nil
            }

            do {
                let value = try self.performMap(element)
                self.activeCount += 1
                return value
            }
            catch let e {
                self.forwardOn(.error(e))
                self.dispose()
                return nil
            }
        }
    }

这个方法会首先加锁,然后执行self.performMap(element)方法,这个方法在子类(FlatMapSink)中有实现,其实就是调用self.selector:

    override func performMap(_ element: SourceElement) throws -> SourceSequence {
        try self.selector(element)
    }

而这个selector其实就是在最初调用Observable.flatMap时传入的closure,所以到这里,myBtn的点击事件就会被转换成了一个事件序列。获取这个返回值之后,on方法会调用subscribeInner方法:

    func subscribeInner(_ source: Observable<Observer.Element>) {
        let iterDisposable = SingleAssignmentDisposable()
        if let disposeKey = self.group.insert(iterDisposable) {
            let iter = MergeSinkIter(parent: self, disposeKey: disposeKey)
            let subscription = source.subscribe(iter)
            iterDisposable.setDisposable(subscription)
        }
    }

在这里,主要是创建了一个MergeSinkIter对象,并让这个MergeSinkIter对象订阅了传入的Observable。根据MergeSinkIter的on方法:

    func on(_ event: Event<Element>) {
        self.parent.lock.performLocked {
            switch event {
            case .next(let value):
                self.parent.forwardOn(.next(value))
            case .error(let error):
                self.parent.forwardOn(.error(error))
                self.parent.dispose()
            case .completed:
                self.parent.group.remove(for: self.disposeKey)
                self.parent.activeCount -= 1
                self.parent.checkCompleted()
            }
        }
    }

我们可以看见它实际就是将事件传递给了parent,也就是MergeSink。但是注意,这里是直接调用的forwardOn方法,所以会直接把事件传递给下游observer,而不会触发performMap方法(当然也没法直接调用on方法,因为二者接收的参数泛型通常是不一样的)。

这样一来,flatMap的完整流程就结束了。

总结一下:
1.通过flatMap创建了一个FlatMap对象,并通过FlatMapSink订阅了上游Observable
2.当上游Observable产生一个事件时FlatMapSink通过调用FlatMap.performMap方法触发传入的closure,产生了新的事件序列。
3.通过MergeSinkIter将新的事件序列的事件传递个下游的observer,从而实现了Event->Observable->Event的传递。

它的流程和RxSwift中大部分操作符的核心逻辑差不多,都是通过持有上游observable,创建一个Sink的子类去订阅这个上游observable,然后将事件传递个下游的observer。只不过因为这里产生的是事件序列,所以又多出了一个MergeSinkIter来处理这个事件序列的事件并进行传递。
当然,今天只是介绍了FlatMap的具体流程,MergeSink的相关细节我们都忽略了,这个我们后面再说。

码字不易,若有错漏,欢迎指正。若有助益,烦请点赞。^ _ ^

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

推荐阅读更多精彩内容