前面介绍了信号,这里介绍一个信号发生器SignalProducer。关于信号发生器的图例可参考自ReactiveCocoa 4 图解之六——信号发生器(SignalProducer)。
冷信号
为什么称SignalProducer为冷信号,看下面例子:
let producer = SignalProducer.init { (observer, _) in
print("新的订阅,启动操作")
observer.send(value: "Hello")
observer.send(value: "World")
}
let subscriber1 = Observer(value: { print("观察者1接收到值 \($0)") })
let subscriber2 = Observer(value: { print("观察者2接收到值 \($0)") })
print("观察者1订阅信号发生器")
producer.start(subscriber1)
print("观察者2订阅信号发生器")
producer.start(subscriber2)
//注意:发生器将再次启动工作
打印结果:
观察者1订阅信号发生器 新的订阅,启动操作 观察者1接收到值 Hello 观察者1接收到值 World 观察者2订阅信号发生器 新的订阅,启动操作 观察者2接收到值 Hello 观察者2接收到值 World
所以称SignalProducer是冷信号,任何一个订阅者/观察者都不会错过任何事件。start方法有关联一个观察者并激活信号的功能。
SignalProducer.empty
let emptyProducer = SignalProducer.empty
let observer = Observer.init(value:{ _ in print("打印值") },
failed: { _ in print("失败调用") },
completed: { _ in print("完成调用") },
interrupted:{ _ in print("阻断调用") })
emptyProducer.start(observer)
打印结果:
完成调用
注意:这里打印的是“完成调用”, 而Signal调用的是interrupted。可能是为了区分Signal是有时序的,SignalProducer是没有时序的。
SignalProducer.never
会返回一个什么都不会发送信号的生成器,同上。
buffer
创建一个时间队列可以回放已经发送的事件 在5.0已移除API
startWithXXX
如startWithSignal, 返回信号开始发送的事件
print("-------------------startWithSignal-------------------")
var v1: String?
SignalProducer.init(value: "Hello")
.on(value: { (text) in
v1 = text
})
.startWithSignal { (signal, disposable) in
print(signal)
print(v1) //nil
}
print(v1)
print("-------------------startWithValues-------------------")
SignalProducer.init(value: "Hello")
.startWithValues { (value) in
print(value)
}
打印结果:
-------------------startWithSignal------------------- ReactiveSwift.Signal
总结起来:startWithCompleted、startWithFailed、startWithInterrupted同startWithValues,只不过只能接受相应的事件
lift
var word = ""
let transform: (Signal) -> Signal = { signal in
word = "Hello"
return signal
}
SignalProducer.init(value: "").lift(transform).startWithValues { (_) in
print(word)
}
会打印:Hello
这个个人觉得较难理解,“提起”对信号发生器进行操作,可理解为所有的原函数都是通过lift去实现的,接用中间信号来实现一系列的信号变换。
map
匹配值转换为新的值
mapError
把收到的error转换为新的Error
filter
过滤不符合条件的值
take
去前几次的值
以上都和热信号相关API使用类似。
observeOn
最后在介绍一个observeOn, 即在指定的调度器上分发事件
let baseProducer = SignalProducer.init(values: ["Hello", "Nice", "to", "meet", "you"])
let completion = {
print("是主线程吗?\(Thread.current.isMainThread)")
}
//主线程
baseProducer.observe(on: QueueScheduler(qos: DispatchQoS.default, name: "test")).startWithCompleted(completion)
//非主线程
baseProducer.startWithCompleted(completion)
打印结果:
是主线程吗?true 是主线程吗?false
相关链接:
Swift 响应式编程FRP使用RAC学习笔记1
Swift 响应式编程FRP使用RAC学习笔记2