使用 Combine 的自定义更改运算符

Custom change operator using Combine

我想创建一个 Combine publisher/subscriber/subscription,它的行为如下:

struct Change<Value> {
  let new: Value
  let previous: Value?
}

let pub = PassthroughSubject<Int, Never>()

let cancellable = pub
  .change()
  .sink { (change: Change<Int>) -> Void in
    print(change)
  }

pub.send(1) // prints Change(new: 1, previous: nil)
pub.send(2) // prints Change(new: 2, previous: 1)
pub.send(3) // prints Change(new: 3, previous: 2)

无法正确实施。我已经制作了自己的 Publisher/Subscription 来包装外部 API 调用和排序,但是当需要保留某些状态时无法提出正确的组合,例如本例中的先前值(我认为这意味着您需要自定义订阅者?)

如果由于某种原因 .change() 语法不可用,也可以使用具有相同语义的替代语法。

您可以像这样从 scanmap 运算符构建 change 运算符:

struct Change<Value> {
    var old: Value?
    var new: Value
}

extension Publisher {
    func change() -> Publishers.Map<Publishers.Scan<Self, (Optional<Self.Output>, Optional<Self.Output>)>, Change<Self.Output>> {
        return self
            .scan((Output?.none, Output?.none)) { (state, new) in
                (state.1, .some(new))
            }
            .map { (old, new) in
                Change(old: old, new: new!)
            }
    }
}

演示:

let pub = PassthroughSubject<Int, Never>()
let ticket = pub
    .change()
    .sink { print([=11=]) }

pub.send(1)
// Output: Change<Int>(old: nil, new: 1)

pub.send(2)
// Output: Change<Int>(old: Optional(1), new: 2)

pub.send(3)
// Output: Change<Int>(old: Optional(2), new: 3)