01. RxSwift源码解读:基本订阅流程

今天带大家解读下订阅发布流程的内部源码。

本系列文章RxSwift使用的版本是:6.2.0

let observable = Observable<Int>.create { (anyObserver) -> Disposable in
            anyObserver.onNext(1)
            anyObserver.onCompleted()
            return Disposables.create ()
        }
        let dispose = observable.subscribe(onNext: { ele in
            print(ele)
        }, onDisposed:  {
            print("dispose")
        })
        dispose.dispose()

上面代码创建一个被观察者,并订阅它,打印序列元素,最后dispose,最后打印:

1
dispose

RxSwift的基本订阅流程涉及到许多类和协议,需要先弄清楚各个类和协议的作用以及它们之间的关系,否则直接看代码容易绕晕。
先上个类图把所有相关类和协议关系理清楚。


类图.jpg

其中蓝色的是类,橙色的协议,黄色的是枚举, 绿色的是结构体。我们一个一个说:

  • 协议:
    ObservableConvertibleType 这是个被观察者的协议,唯一的一个协议方法是asObservable(), 表示可以转换成被观察者。
    ObservableType: 继承自ObservableConvertibleType,可以创建被观察者,任何被观察者类需要遵循此协议。在extension中定义了一个create函数用来创建被观察者,两个subscribe函数用来订阅观察者。
    Disposable 订阅取消的接口,只有一个disposes协议方法,用来释放相关资源。
    Cancelable 继承自Disposable,有一个isDisposed协议方法,表示是否已释放资源。
    ObserverType 表示序列发布者,可以发送序列。其中有几个我们比较熟悉的方法:onNext onError onCompleted 这三个会调用on方法,on只是个协议方法,看下代码就明白了:
/// Supports push-style iteration over an observable sequence.
public protocol ObserverType {
    /// The type of elements in sequence that observer can observe.
    associatedtype Element

    @available(*, deprecated, message: "Use `Element` instead.")
    typealias E = Element

    /// Notify observer about sequence event.
    ///
    /// - parameter event: Event that occurred.
    func on(_ event: Event<Element>)
}

/// Convenience API extensions to provide alternate next, error, completed events
extension ObserverType {
    
    /// Convenience method equivalent to `on(.next(element: Element))`
    ///
    /// - parameter element: Next element to send to observer(s)
    public func onNext(_ element: Element) {
        self.on(.next(element))
    }
    
    /// Convenience method equivalent to `on(.completed)`
    public func onCompleted() {
        self.on(.completed)
    }
    
    /// Convenience method equivalent to `on(.error(Swift.Error))`
    /// - parameter error: Swift.Error to send to observer(s)
    public func onError(_ error: Swift.Error) {
        self.on(.error(error))
    }
}

onNext onError onCompleted都调用了on方法,它还包含一个关联类型,Element可以认为是个范型,表示元素的类型。


  • Observable Producer AnonymousObservable 这三个都是被观察者,依次继承的关系,Observable遵循ObservableType, Producer 继承Observable,AnonymousObservable继承了Producer。Producer,实现subscribe,run方法, AnonymousObservable实现了run方法。

SinkAnonymousObservableSink 这两个算是是整个流程的核心类,对消息订阅发送进行管理。其中Sink 遵循Dispsable,包含两个属性ObserverType和Cancelable,这是整个类图唯一的两个组合关系,其他类的属性都是闭包,AnonymousObservableSink继承了Sink,同时遵循了ObserverType,也就是说AnonymousObservableSink 包括一个ObserverType类型的属性(由父类继承而来)同时又遵循了ObserverType协议,这让我想起的设计模式中的装饰模式,其实整个Rx框架还有很多类型的Sink。

ObserverBase AnonymousObserver, 遵循了Dispose和 ObserverType,这两个是观察者,继承关系。

  • 结构体 唯一的一个结构体AnyObserver,遵循了ObserverType 这个好像也是观察者,实际上是序列的发送者,用户通过它来调用onNext 等方法发送序列,被观察者如Observable通过AnyObserver发送序列,而AnonymousObserver对象负责接收序列。

  • 枚举 Event 表示序列事件,包含next(Element) error(Swift.Error) onCompleted三个case。
    终于介绍完了所有的类和协议。
    接下来需要说到三个重要的闭包:

  1. 创建被观察者的闭包即我们一开始的代码中的
{ (anyObserver) -> Disposable in
            anyObserver.onNext(1)
            anyObserver.onCompleted()
            return Disposables.create ()
        }

它的类型是(AnyObserver)-> Disposeable 在订阅时会执行这个闭包,并且用anyObserver发送序列。

  1. 订阅时的onNext的闭包:
  (onNext: { ele in
            print(ele)
        }, onDisposed:  {
            print("dispose")
        })

这样发送next序列时,会调用这个闭包。

  1. 第三个闭包在subscribe方法的内部,我们进去看一下:
let observer = AnonymousObserver<Element> { event in
                
                #if DEBUG
                    synchronizationTracker.register(synchronizationErrorMessage: .default)
                    defer { synchronizationTracker.unregister() }
                #endif
                
                switch event {
                case .next(let value):
                    onNext?(value)
                case .error(let error):
                    if let onError = onError {
                        onError(error)
                    }
                    else {
                        Hooks.defaultErrorHandler(callStack, error)
                    }
                    disposable.dispose()
                case .completed:
                    onCompleted?()
                    disposable.dispose()
                }
            }

这里创建了一个AnonymousObserver, 也就是观察者,创建时将闭包作为初始化参数,闭包中调用了第二个闭包onNext。所以第一个闭包和第三个闭包是如果关联的呢?也就是订阅的时候如何调用第一个闭包,第一个闭包再调用到第三个闭包?这是整个流程的关键?
我们再上个流程图看看整个订阅发布的流程。


订阅流程图.jpg

根据这个图再结合源码一步一步分析下流程。

  1. 首先通过Observable的 create方法创建序列,ObservableType extension 提供了实现,而Observable遵循了ObservableType协议,可以看到代码创建了AnonymousObservable(subscribe), 并把闭包传进入,而AnonymousObservable类内部持有了这个闭包。
public static func create(_ subscribe: @escaping (AnyObserver<Element>) -> Disposable) -> Observable<Element> {
        return AnonymousObservable(subscribe)
    }

final private class AnonymousObservable<Element>: Producer<Element> {
    typealias SubscribeHandler = (AnyObserver<Element>) -> Disposable

    let _subscribeHandler: SubscribeHandler

    init(_ subscribeHandler: @escaping SubscribeHandler) {
        self._subscribeHandler = subscribeHandler
    }
}
  1. 用返回的AnonymousObservable对象,调用subscribe方法,这个方法在ObservableType 扩展中,看一看subscribe主要代码:
        let observer = AnonymousObserver<Element> { event in
     
                #if DEBUG
                    synchronizationTracker.register(synchronizationErrorMessage: .default)
                    defer { synchronizationTracker.unregister() }
                #endif
                
                switch event {
                case .next(let value):
                    onNext?(value)
                case .error(let error):
                    if let onError = onError {
                        onError(error)
                    }
                    else {
                        Hooks.defaultErrorHandler(callStack, error)
                    }
                    disposable.dispose()
                case .completed:
                    onCompleted?()
                    disposable.dispose()
                }
            }
            return Disposables.create(
                self.asObservable().subscribe(observer),
                disposable
            )

先包装一个AnonymousObserver,即观察者,把闭包作为初始化方法的参数传进去,并赋值给它的_eventHandler属性,跟AnonymousObservable比较类似,不要与AnonymousObservable弄混了。一个是观察者,一个是被观察者。
闭包的代码中根据事件类型调用 onNext 或 onError 或 onCompleted闭包,这个是最后一步的调用, 顺便提一下处理.error 和 .completed 事件时会调用disposable.dispose(),说明这两个事件发生后会取消订阅,回收资源,之后无法再发送序列了。

  1. 接着看return Disposables.create( self.asObservable().subscribe(observer), disposable )
    这里会创建Disposables,它传入两个disposeable,这里先不讲dispose,看第一个参数, 通过asObservable 调用subscribe,并把刚刚创建的observer传入,asObservable我们可以通过上面的类图看到它是ObservableConvertibleType的协议,Observable 必然实现了这个协议,实际上是返回自身。自身的类型其实是AnonymousObservable,所以就是通过self调用subscribe,subscribe在父类Producer实现了。
  2. 接着看Producer 的subscribe代码:
override func subscribe<Observer: ObserverType>(_ observer: Observer) -> Disposable where Observer.Element == Element {
        if !CurrentThreadScheduler.isScheduleRequired {
            // The returned disposable needs to release all references once it was disposed.
            let disposer = SinkDisposer()
            let sinkAndSubscription = self.run(observer, cancel: disposer)
            disposer.setSinkAndSubscription(sink: sinkAndSubscription.sink, subscription: sinkAndSubscription.subscription)

            return disposer
        }
        else {
            return CurrentThreadScheduler.instance.schedule(()) { _ in
                let disposer = SinkDisposer()
                let sinkAndSubscription = self.run(observer, cancel: disposer)
                disposer.setSinkAndSubscription(sink: sinkAndSubscription.sink, subscription: sinkAndSubscription.subscription)

                return disposer
            }
        }
    }

CurrentThreadScheduler是线程派发,以后再说线程派发,调用schedule,在schedule会执行传入的闭包,
image.png

所以最后会走到闭包中,关注下闭包的代码:调用了 self.run, 把参数observer 传进去, observer是外面创建的AnonymousObserver对象,我们整个流程只会有一个AnonymousObserver和一个AnonymousObservable其他地方看到的observer都是传进去的,所以看到observer简单想到是最开始创建的AnonymousObserver就行了。

  1. 现在到run方法了,AnonymousObservable实现了run方法:
override func run<Observer: ObserverType>(_ observer: Observer, cancel: Cancelable) -> (sink: Disposable, subscription: Disposable) where Observer.Element == Element {
        let sink = AnonymousObservableSink(observer: observer, cancel: cancel)
        let subscription = sink.run(self)
        return (sink: sink, subscription: subscription)
    }
  1. 这里比较关键:创建了AnonymousObservableSink对象,Sink翻译做“业务下沉”,表示这个类是管理者或者专门处理业务的。将observer和cancel传进去。observer还是通过调用链传来的。接着调用AnonymousObservableSink的run方法:sink.run(self)
  2. 看一下AnonymousObservableSink的run方法:
    func run(_ parent: Parent) -> Disposable {
        return parent._subscribeHandler(AnyObserver(self))
    }

Parant 是 sink.run(self) 的self, self实际上是AnonymousObservable对象,就是唯一的被观察者,是用户通过create创建的。

  1. 所以接下来调用了AnonymousObservable对象的_subscribeHandler,同时把AnyObserver(self)最为参数传入,_subscribeHandler是个闭包,我们可以回到第1步看看。_subscribeHandler是最开始 create 被观察者传入的闭包,所以到这一步才开始执行第1步创建的闭包,这里还有个关键点:AnyObserver(self), 创建了一个AnyObserver并把self作为参数传入,self是 AnonymousObservableSink对象啊,记住了后面会比较绕。 打开AnyObserver看看这个初始化方法:
    /// Construct an instance whose `on(event)` calls `observer.on(event)`
    ///
    /// - parameter observer: Observer that receives sequence events.
    public init<Observer: ObserverType>(_ observer: Observer) where Observer.Element == Element {
        self.observer = observer.on
    }

是这个没问题,通过类图可以看到AnonymousObservableSink遵循了ObserverType,把on方法赋值给observer。

  1. self.observer = observer.on, 把AnonymousObservableSink的on 方法复制给observer 这里的observer可不是AnonymousObserver对象,它是个闭包,类型是(Event<Element>) -> Void。诶,我是谁!我在哪里!😮💨,我们回到执行_subscribeHandler的地方。
  2. 第8步说到执行_subscribeHandler,也就是我们最开始create Observable的闭包:
      let observable = Observable<Int>.create { (anyObserver) -> Disposable in
            anyObserver.onNext(1)
            anyObserver.onCompleted()
            return Disposables.create()
       }
  1. 当代码执行anyObserver.onNext(1), anyObserver是刚刚第8步创建的哦, AnyObserver没有找到方法,它其实在ObserverType extension中(AnyObserver 遵循了ObserverType):
/// Convenience API extensions to provide alternate next, error, completed events
extension ObserverType {
    
    /// Convenience method equivalent to `on(.next(element: Element))`
    ///
    /// - parameter element: Next element to send to observer(s)
    public func onNext(_ element: Element) {
        self.on(.next(element))
    }

执行了on方法

  1. AnyObserver有实现了 on 方法协议:
public func on(_ event: Event<Element>) {
        return self.observer(event)
    }
  1. 调用self.observer(event), 这个observer 在第9步(self.observer = observer.on)由AnonymousObservableSink.on 赋值的闭包。所以解下来调用AnonymousObservableSink.on方法。

  2. 我们看看on方法实现:

func on(_ event: Event<Element>) {
        #if DEBUG
            self._synchronizationTracker.register(synchronizationErrorMessage: .default)
            defer { self._synchronizationTracker.unregister() }
        #endif
        switch event {
        case .next:
            if load(self._isStopped) == 1 {
                return
            }
            self.forwardOn(event)
        case .error, .completed:
            if fetchOr(self._isStopped, 1) == 0 {
                self.forwardOn(event)
                self.dispose()
            }
        }
    }

我们看到AnonymousObservableSink 类的 run 方法和on 方法在整个流程中至关重要,run方法调用了订阅的handler,而on 方法处理了事件。这说明AnonymousObservableSink类处理主要业务逻辑,是整个流程的核心。

  1. 最终会调用self.forwardOn(event), 这个forwardOn是在父类Sink定义的,然后跳到forwardOn看看:
    final func forwardOn(_ event: Event<Observer.Element>) {
        #if DEBUG
            self._synchronizationTracker.register(synchronizationErrorMessage: .default)
            defer { self._synchronizationTracker.unregister() }
        #endif
        if isFlagSet(self._disposed, 1) {
            return
        }
        self._observer.on(event)
    }
  1. 然后调用 self._observer.on(event), 这个_observer 是在第6步创建AnonymousObservableSink对象时作为初始化参数赋值的。这个observer就是唯一的AnonymousObserver对象,还知道它是什么时候创建的吗?所以我们去AnonymousObserver找下on方法,没找到,去父类ObserverBase找到了:
    case .next:
            if load(self._isStopped) == 0 {
                self.onCore(event)
            }
  1. 接着调用onCore,在AnonymousObserver里:
    override func onCore(_ event: Event<Element>) {
        return self._eventHandler(event)
    }

终于绕出来了,调用self._eventHandler(event), _eventHandler 现在还记得是啥吗?啥时候被赋值的?_eventHandler是在第2步创建AnonymousObserver时被赋值的。所以接着调用第2步的代码:case .next(let value): onNext?(value) 调用onNext就是我们订阅时传入的闭包。最后打印1, 希望大家能看明白,下一篇文章会解析dispose流程。

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

推荐阅读更多精彩内容