Swift见解

ReactiveCocoa - Swift 解读系列二:Ober

2017-11-17  本文已影响10人  BennyLoo

在上一篇文章中,讲到了RAS中Signal的创建和使用,展示了从Signal创建到绑定观察者以及发送的过程。本文将介绍关于SignalOberser之间的解绑关系。

上篇文章开始的示意图中,我们知道,除了本身的发送器,所有的观察者Oberser对象存在Signal的Bag(之前已经讲过,这是一个集合结构体,我们将它看作一个数组类似的东西即可,它被当成Signal状态的一个关联之放在Signal中)中了。如果我们要将一个观察者加入到Signal,可以使用Signal的绑定函数:

    public func observe(_ observer: Observer) -> Disposable? {
        return core.observe(observer)
    }

这个函数调用Signal中Core类的func observe(oberser)函数,具体的实现看下图:

Core 中绑定观察者
函数中,首先会判断Signal本身的状态State(这是一个枚举,包括aLive和terminating以及terminated三种状态),如果处于aLive(活跃)状态的时候,则会将观察者加入到Signal的bag中,完成绑定。否则,观察者observer将发送一个中断事件。

注意观察者observer在加入到bag之后,Bag 的token会发生相应的变化,这个token中包含的就是observer在Bag中的类似索引机制,Bag通过操作token来控制内部的observer。如果变化后的token是有效的,那么将会创建observer对应的Disposable。并实现Disposable的一个回调函数action: ()->(Void).其内容正如我们看到的,在bag内部将observer删除,这样就是消除了observerSignal信号消息的观察。最后整个函数将这个Disposable返回。

也就是说Disposable有一套内部的逻辑,在合适的时候,调用其action函数,从而将observer解除。我们再看看Disposable到底是什么鬼。

Disposable源码中:

Disposable
包含一个Bool变量 var isDisposed: Bool { get }和一个清理函数 func dispose()。可以说很简单,我们基本上可以猜测到,isDisposed存储的是Disposable本身是否被(在Oberser 被清理出Bag后这个状态将会为true);而dispose()就是将Oberser的绑定解除的函数。

Signal并没有直接遵循这个协议,而是会使用遵循了下面这个协议的几个类:

这个类是Disposable类族中的一个基本类,也是一个internal类,使用者不会直接使用到它。它只负责创建一个Disposable的空类,也即是没有消费过的Disposable

创建一个已经被消费的Disposable。它除了能知道一个Disposable的状态之外没有其他作用。

AnyDisposable是创建一个有效的Disposable最主要的方式。我们在建立一个SignalOberser绑定的时候,其中返回的Disposable就是一个AnyDisposable对象。

综合类型Disposable,它的内部又一个包含AnyDisposable的bag。并重载了+= 运算符。使用它可以持续的添加其他的AnyDisposable加入到bag中。

Disposable的封包。支持对AnyDisposable和CompositeDisposable的封装。

一系列的Disposable

这篇文章主要讲解Oberser如何退出观察的,也即是重点讲解下AnyDisposable。以下是AnyDisposable的源码:
`AnyDisposable`
AnyDisposable继承自Disposable。内包含一个ActionDisposable,这个类类似于SignalCore.RAS大量的采用了内嵌类的设计模式。

除了ActionDisposable之外,AnyDisposable含有三个初始化函数。我们主要讲下 public init(_ action: @escaping () -> Void){}.因为在 OberserSignal绑定的时候,就是使用了这个函数。

action参数是一个无参函数,它是一个尾随闭包。最终我们看到通过ActionDisposable将这个闭包给了ActionDisposable的action。而在func dispose()的执行中,调用了这个闭包。也即是我们在使用AnyDisposable进行解绑的时候,实际调用了我们在绑定时候传入的一个闭包:

`AnyDisposable`的回调函数
这里看到,当ActionDisposable执行dispose()的时候实际上做的事情是执行了Signal的removOberser()函数,这个函数的实现:

将对应的OberserSignal的Bag(case alive(Bag<Observer>, hasDeinitialized: Bool) )中清除了。最终达到停止OberserSignal绑定或者说订阅。

Signal解绑Oberser的实例
实例

打印结果:


打印结果

从结果中我们可以看到,在调用dispose之前,可以正常观察到发送器发射出来的信息。并且看到isDisposed属性为false。 执行disp.dispose() 之后,isDisposed属性为true,同时也不再接受发射器发送的信号了。

关于Signal解绑Oberser的原理就是这样了。

上一篇下一篇

猜你喜欢

热点阅读