ReactiveSwift源码解析(四) Signal中的静态属性静态方法以及面向协议扩展
上篇博客我们聊了Signal的几种状态、Signal与Observer的关联方式以及Signal是如何向关联的Observer发送事件的。本篇博客继续上篇博客的内容,来聊一下Signal类中静态的never和empty计算属性以及pipe()静态方法。然后再聊聊Signal中的面向协议编程中的面向协议扩展。在Signal中,只要是对Signal的扩展都是加在了Signal所实现的协议中,稍后会进行介绍。
一、Signal中获取实例的静态计算属性
在本篇博客的第一部分我们先来看看Signal类中的两个属性,一个是never,另一个是empty。之所以将这两个计算属性放在一块,是因为这两个静态计算属性都是用来获取Signal实例的。但是所获取实例的功能不同。解析来我们就来看一下never以及empty的实现和使用方式。
1. never
下方是never是代码实现,实现比较简单。其中就是调用了Signal的构造器,但是没有在构造器的尾随闭包中做任何事情。通过该计算属性获取的Signal对象,不会获取到内置的Observer对象,从而Signal的对象持有者是不能对Signal所关联的观察者发送事件的。

下方是ReactiveSwift官方关于never的测试用例以及输出结果。从运行结果中来看,所关联的Observer对象是不会收到来自Signal的任何消息的。

2、empty
聊完never,接下来我们来看一下Signal的静态计算属性empty的实现以及执行方式。empty的使用方式与never差不多,empty形式的Signal在管理Observer时,只会执行该Observer的interrupted事件。解析下来我们就来看一下empty的实现方式。
我们先看一下empty的使用方式,下方这段代码就是ReactiveSwift官方的empty使用的示例,以及该示例的输出结果。我们从Signal的静态计算属性empty中获取Signal是实例。然后在关联Observer时,都会执行Observer的interrupted事件的闭包体。

接下来我们来看一下empty的实现方式,代码比较简单,比never就多了一个observer.sendCompleted()方法。在之前我们聊Observer时,我们知道sendCompleted()就是执行观察者的Event.completed事件。sendComplented()会执行Observer中的Action,并带有.completed事件。

下方这个代码段中Observer初始化时的尾随闭包,就是observer.sendComplented()方法所执行的内容。而在这个尾随闭包中,我们看到有一个event.isTerminating的判断,当是.failed、.completed 和 .interrupted事件时event.isTerminating的值都是true。所以该if块中就是empty要执行的方法。
在if语句块中,核心的内容就是修改当前Single的SignalState。当调用Signal的构造器时,SignalState默认是SignalState.alive(AliveState()),而if语句块中就负责将该状态修改成TerminatingState状态,如下所示。
在关联Observer时,会用到TerminatingState状态,下方会给出介绍。

下方代码段就是Signal关联Observer是所调用的方法。从下方代码不难看出,当Signal处于非活跃状态.alive时,token的值就是nil,当token未赋值时,就会执行所关联对象Observer的sendInterrupted()方法,向所关联的Observer发送.interrupted事件,从而执行Observer的interrupted的闭包体。至此,empty的相关内容就解析完了。

二、Signal的静态方法pipe()
Signal中的静态方法pipe()本质上就是一个便利构造器,该便利构造器返回的参数是一个元组,其不仅仅返回一个Signal的实例,而且返回Signal用于发送事件的内置observer对象。pipe()是获取Signal实例的主要方式,接下来我们就来看一下pipe的使用方式以及pipe()的内部实现。
1、pipe()的使用示例
下方这个pipe()的应用示例是从官网Demo的基础上修改的,下方是对该段代码的介绍:
首先通过Signal的pipe()静态方法可以获取一个Signal实例以及该实例所持有的Observer对象,也就下方元组中的(signal, sendMessage)。
紧接着我们创建两个Observer对象,并且给出Value事件所执行的闭包体。我们将这两个Observer的实例命名为subscriber1和subscriber2。
然后我们将subscriber1添加到signal中,在signal调用observe()方法添加Observer时,会返回一个ActionDisposable类型的对象,我们可以使用该对象移除观察者。添加完毕后,我们就调用sendMessage的send(value)方法,发送value事件,然后观察者subscriber1所对应的Value事件闭包体就会得到执行。
我们以同样的方将subscriber2添加到Signal中,然后通过是调用sendMessage的send(value)方法,发送value事件。我们就可以看到subscriber1和subscriber2这两个观察者的Value闭包就会执行。
然后我们调用actionDisposable对象的dispose()方法,将subscriber1从Signal的Bag中移除掉。移除后subscriber1就不会收到来自Signal的事件了。actionDisposable对象的dispose()方法稍后会介绍到。
我们再次调用sendMessage的send(value)方法,subscriber1的Value事件的闭包就不会被执行。

2、pipe()的代码实现
pipe()的代码实现比较简单,就是通过元组将Signal的对象以及Observer的对象进行返回即可,下方就是pipe()的代码实现。因其实现比较简单,在此就不做过多赘述了。

3、ActionDisposable的代码实现
接下来我们来解析一下ActionDisposable的代码实现,在每次观察者Observer与Signal调用observe()方法进行关联时都会返回一个ActionDisposable对象,该对象可以是对应的观察者取消对Signal信号的观察。接下来我们就来看一下ActionDisposable的具体实现。
下方就是ActionDisposable类的实现,ActionDisposable的代码实现比较简单,本质上就是通过构造器接收一个闭包,然后将这个尾随闭包赋值给action变量,然后在dispose()方法中去调用该action闭包。

下方就是在关联Observer和Signal的observe()方法中实例化ActionDisposable的相关代码。下方的大红框中就是ActionDisposable方法中action的闭包体,也是dispose()方法中主要执行的代码。在改闭包中所执行的目的也是比较单一的,其中主要做了一件事情,就是根据token从Signal活跃状态的Bag中移除相应的Observer,换句话说就是移除观察者。

三、Signal的可扩展性
在本篇博客的最后一部分,想聊一下Signal的可扩展性设计。对Signal功能的扩展,主要使用了面向协议扩展的形式。主要就是是Signal实现SignalProtocol,然后我们对 SignalProtocol这个协议进行扩展,而不是对Signal这个类本身进行扩展。所以此处我们称之为“面向协议扩展”,对SignalProtocol这个协议进行扩展后,因为Signal这个类遵循SignalProtocol,所以Signal也会拥有SignalProtocol所扩展的功能。
下方截图中就是SignalProtocol的实现以及相应的扩展。从下方代码中我们可以看到,Signal类的大部分核心功能都是通过SignalProtocol的协议扩展而拥有的。SignalProtocol有好多扩展,本篇博客就不细说了,下篇博客我们找一些比较核心的SignalProtocol的扩展拿出来聊聊。

今天博客就先到这儿,下篇博客我们会对ReactiveSwift中的SignalProtocol的延展的实现进行介绍。
上述代码github分享地址:https://github.com/lizelu/TipSwiftForRac
ReactiveSwift源码解析(四) Signal中的静态属性静态方法以及面向协议扩展的更多相关文章
- ReactiveCocoa源码解析(四) Signal中的静态属性静态方法以及面向协议扩展
上篇博客我们聊了Signal的几种状态.Signal与Observer的关联方式以及Signal是如何向关联的Observer发送事件的.本篇博客继续上篇博客的内容,来聊一下Signal类中静态的ne ...
- ReactiveSwift源码解析(三) Signal代码的基本实现
上篇博客我们详细的聊了ReactiveSwift源码中的Bag容器,详情请参见<ReactiveSwift源码解析之Bag容器>.本篇博客我们就来聊一下信号量,也就是Signal的的几种状 ...
- ReactiveSwift源码解析(七) Signal的CombineLatest的代码实现
本篇博客我们就来聊一下combineLatest()的使用以及具体的实现方式.在之前的<iOS开发之ReactiveCocoa下的MVVM>的博客中我们已经聊过combineLatest( ...
- ReactiveSwift源码解析(五) SignalProtocol的observe()、Map、Filter延展实现
上篇博客我们对Signal的基本实现以及Signal的面向协议扩展进行了介绍, 详细内容请移步于<Signal中的静态属性静态方法以及面向协议扩展>.并且聊了Signal的所有的g功能扩展 ...
- ReactiveCocoa源码解析(三) Signal代码的基本实现
上篇博客我们详细的聊了ReactiveSwift源码中的Bag容器,详情请参见<ReactiveSwift源码解析之Bag容器>.本篇博客我们就来聊一下信号量,也就是Signal的的几种状 ...
- Mybatis源码解析(四) —— SqlSession是如何实现数据库操作的?
Mybatis源码解析(四) -- SqlSession是如何实现数据库操作的? 如果拿一次数据库请求操作做比喻,那么前面3篇文章就是在做请求准备,真正执行操作的是本篇文章要讲述的内容.正如标题一 ...
- Sentinel源码解析四(流控策略和流控效果)
引言 在分析Sentinel的上一篇文章中,我们知道了它是基于滑动窗口做的流量统计,那么在当我们能够根据流量统计算法拿到流量的实时数据后,下一步要做的事情自然就是基于这些数据做流控.在介绍Sentin ...
- Spring5源码解析-Spring框架中的单例和原型bean
Spring5源码解析-Spring框架中的单例和原型bean 最近一直有问我单例和原型bean的一些原理性问题,这里就开一篇来说说的 通过Spring中的依赖注入极大方便了我们的开发.在xml通过& ...
- ReactiveSwift源码解析(一) Event与Observer代码实现
ReactiveCocoa这个框架是做什么用的本篇博客就不做过多赘述了,什么是"响应式编程"也不多聊了,自行Google吧.本篇博客的主题是解析ReactiveCocoa框架中的核 ...
随机推荐
- Ubuntu物理机中解决VirtualBox虚拟机无法连接USB设备的问题
本文由荒原之梦原创,原文链接:http://zhaokaifeng.com/?p=611 问题描述: 在安装完VirtualBox的USB控制器扩展(关于在VirtualBox中安装USB控制器扩展的 ...
- sql server 高可用故障转移(3)
虚拟磁盘创建 前面我们已经搭了域和两台sql 服务器, 下面我们准备让DC域服务器除了担当域控制器外,还行使另一个职能:充当集群共享存储. 集群共享存储是由群集内的每个节点都能共同访问的一个存储设备, ...
- mac里用PyCharm中引用MySqlDB始末
本来想用java来连数据库,然后调用python的,后来想了想,反正是个实验性质的小工程何必搞的这么复杂.直接全部python就好了,于是就为这个想法填了一晚上的坑. 装好了PyCharm的CE版,然 ...
- TestNG 自动化测试入门教程--典型示例
TestNG介绍 TestNG是Java中的一个测试框架, 类似于JUnit 和NUnit, 功能都差不多, 只是功能更加强大,使用也更方便 Java中已经有一个JUnit的测试框架了. Tes ...
- APNs 推送原理及问题
http://bbs.csdn.net/topics/390461996 在 iOS 平台上,大部分应用是不允许在后台运行并连接网络的.在应用没有被运行的时候,只能通过 Apple Push Noti ...
- Spring_boot简单操作数据库
Spring_boot搭配Spring Data JPA简单操作数据库 spring boot 配置文件可以使用yml文件,默认spring boot 会加载resources目录的下的applica ...
- 文本分布式表示(二):用tensorflow和word2vec训练词向量
看了几天word2vec的理论,终于是懂了一些.理论部分我推荐以下几篇教程,有博客也有视频: 1.<word2vec中的数学原理>:http://www.cnblogs.com/pegho ...
- H5单张、多张图片上传
前言 今天我们聊一聊图片上传,单张Or多张 ,如今,各大图片上传插件数不胜数,例如:Jquery的 verupload.js,jQuery File Upload.Uploadify.jQuery.f ...
- 重磅!!!微软发布.NET Core 2.2
我们很高兴地宣布发布.NET Core 2.2.它包括对运行时的诊断改进,对ARM32 for Windows和Azure Active Directory for SQL Client的支持.此版本 ...
- 微信小程序中placeholder的样式
通常,现代浏览器大多支持::placeholder选择器,用于设置placeholder的样式,但是在微信小程序中并不支持这种方式,而是提供了一个专门的属性(placeholder-class)来处理 ...