本文我们着重来分析一种特殊的 Publisher —— Subject。与之前我们讨论的 Publisher 不同的是,Subject 的最大特点就是可以手动发送数据,本文将讨论 Subject 的定义以及基本用法。
一、Subject 的定义
public protocol Subject : AnyObject, Publisher { func send(_ value: Self.Output) func send(completion: Subscribers.Completion<Self.Failure>) func send(subscription: Subscription) }
从定义可以看到,Subject 继承于 Publisher 协议并且暴露了 3 个 send 方法,我们可以通过调用 send 方法来手动发送 Output 数据以及 Completion 事件。
二、Combine 内置的 Subject
Combine 为我们内置了两种 Subject —— PassthroughSubject 与 CurrentValueSubject,它们的区别主要在于是否会对收到的数据进行保留。
PassthroughSubject
PassthroughSubject 通过 send 发送数据或事件给下游的 Publisher 或 Subscriber, 不会对接收到的数据进行保留。
import Combine let subject = PassthroughSubject<String, Never>() // 创建PassthroughSubject // 订阅 let subscription = subject.sink(receiveCompletion: { _ in print("receiveCompletion") }, receiveValue: { value in print(value) }) // 发送数据 subject.send("hello") subject.send("world") subject.send(completion: .finished) /* 输出: hello world receiveCompletion */
CurrentValueSubject
与 PassthroughSubject 不同,CurrentValueSubject 会保留一个最后的数据,并在被订阅时将这个数据发送给下游的 Publisher 或 Subscriber。
import Combine let subject = CurrentValueSubject<String, Never>("hi") // 创建CurrentValueSubject,需要给它赋一个初值 // CurrentValueSubject有value属性 print(subject.value) // 发送数据 subject.send("hello") print(subject.value) subject.send("world") print(subject.value) // 订阅 let subscription = subject.sink { value in print(value) } /* 输出: hi hello world world */