Combine Publisher详解

摘自《SwiftUI和Combine编程》---《Publisher 和常见 Operator》

Publisher 详解

Publisher 在接收到订阅,并且接受到请求要求提供若干个事件后,才开始对外发布事件。接受订阅和接受请求,也是 Publisher 生命周期的一部分。

Empty

Empty<Int, SampleError>()

是一个最简单的 Publisher,它只在被订阅的时候发布一个完成事件 (receive finished)。这个 publisher 不会输出任何的 output 值,只能用于表示某个事件已经发生。

/// A publisher that never publishes any values, and optionally finishes immediately.
///
/// You can create a ”Never” publisher — one which never sends values and never finishes or fails — with the initializer `Empty(completeImmediately: false)`.
public struct Empty<Output, Failure> : Publisher, Equatable where Failure : Error

Just

Just(1)

表示一个单一的值,在被订阅后,这个值会被发送出去,紧接着是 finished

/// A publisher that emits an output to each subscriber just once, and then finishes.
///
/// You can use a ``Just`` publisher to start a chain of publishers. A ``Just`` publisher is also useful when replacing a value with ``Publishers/Catch``.
public struct Just<Output> : Publisher

Publisher.Sequence

如果我们对一连串的值感兴趣的话,可以使用的是 Publishers.Sequence。顾名思义,Publishers.Sequence 接受一个 Sequence:它可以是一个数组,也可以是一个 Range。在被订阅时,Sequence 中的元素被逐个发送出来

Publisher.Sequence<[Int], Never>(sequence: [1,2,3])
//等价于
[1,2,3].publisher

常见 Operator

map

元素变形:map 操作可以为我们完成事件中数据类型的转换

[1,2,3].publisher.map { $0*2 }

compactMap

将 map 结果中那些 nil 的元素去除掉,这个操作通常会“压缩”结果,让其中的元素数减少,这也正是其名字中 compact 的来源

flatMap

展平 降维

map 及 compactMap 的闭包返回值是单个的 Output 值。而与它们不同,flatMap 的变形闭包里需要返回一个 Publisher。也就是说,flatMap 将会涉及两个 Publisher:一个是 flatMap 操作本身所作用的外层 Publisher,一个是 flatMap 所接受的变形闭包中返回的内层 Publisher。flatMap 将外层 Publisher 发出的事件中的值传递给内层 Publisher,然后汇总内层 Publisher 给出的事件输出,作为最终变形后的结果。

check("Flat Map 1") {
    [[1, 2, 3], ["a", "b", "c"]]
        .publisher
        .flatMap {
            $0.publisher
        }
}
// Output:
// 1 2 3 a b c

/**
在被订阅后,这个外层 Publisher 会发送两个 Output 事件 (两个事件的值分别是 [1, 2, 3] 和 ["a", "b", "c"]),每个事件的值被 flatMap 传递到内层,并通过 $0.publisher 生成新的 Publisher 并返回。内层 Publisher 实际上是 [1, 2, 3].publisher 和 ["a", "b", "c"].publisher,它们发送的值将被作为 flatMap 的结果,被“展平” (flatten) 后发送出去。
*/
check("Flat Map 2") {
    ["A", "B", "C"]
        .publisher
        .flatMap { letter in
            [1, 2, 3]
                .publisher
                .map { "\(letter)\($0)" }
        }
}
// Output:
// A1 A2 A3 B1 B2 B3 C1 C2 C3

/**
外层 ["A", "B", "C"] 里的每个元素,作为 flatMap 变形闭包的输入,参与到了内层 Publisher 的 map 计算。内层 Publisher 逐次使用 [1, 2, 3] 里的元素,和输入进来的 letter (也就是 “A”, “B” 和 “C”) 进行拼接后作为新事件发出。
*/

reduce

操作完发消息

将数组中的元素按照某种规则进行合并,并得到一个最终的结果。

与 scan 区别:当序列中值耗尽时,它将发布 finished。而经过 reduce 变形后,新的 Publisher 只会在接到上游发出的 finished 事件后,才会将 reduce 后的结果发布出来。而紧接这个结果,则是新的 reduce Publisher 的结束事件。

scan

每一步都发消息

场景:在某个下载任务执行期间,接受 URLSession 的数据回调,将已接收到的数据量做累加来提供一个下载进度条的界面。

同 reduce,但是会记录中途每一步的过程。

与 reduce 区别:一边进行重复操作,一边将每一步中间状态发送出去。


源码类似: 可直接添加到 Sequence extension 中

extension Sequence {
    public func scan<ResultElement>(_ initial: ResultElement, _ nextPartialResult: (ResultElement, Element) -> ResultElement) -> [ResultElement] {
        var result: [ResultElement] = []
        for x in self {
            result.append(nextPartialResult(result.last ?? initial, x))
        }
        return result
    }
}

removeDuplicates

移除连续出现的重复事件值

removeDuplicates 经常被用来减少那些非常消耗资源的操作,比如由事件触发造成的网络请求或者图片渲染。如果当作为源头的数据没有改变时,所预期得到的结果也不会变化的话,那么就没有必要去重复这样操作。在源头将重复的事件移除,可以让下游的事件流也变得简单。

check("Remove Duplicates") {
    // removeDuplicates 会处理事件源
    ["S", "Sw", "Sw", "Sw", "Swi",
     "Swif", "Swift", "Swift", "S"] 
        .publisher
        .removeDuplicates()
}
// Output:
// subscription: (["S", "Sw", "Swi", "Swif", "Swift", "S"])

错误处理

Fail

Fail 这个内建的基础 Publisher,它所做的事情就是在被订阅时发送一个错误事件.

Fail<Int, SampleError>(error: .sampleError)

public enum SampleError: Error {
    case sampleError
}

mapError

如果 Publisher 在出错时发送的是 SampleError,但订阅方声明只接受 MyError 时,就算实际上 Publisher 只发出 Output 值而从不会发出 Failure 值,我们也无法使用这个 Subscriber 去接收一个类型不符的 Publisher 的事件。

在这种情况下,我们可以通过使用 mapError 来将 Publisher 的 Failure 转换成 Subscriber 所需要的 Failure 类型

Fail<Int, SampleError>(error: .sampleError)
    .mapError { _ in
        myError.myError
    }

抛出错误

Combine 为 Publisher 的 map 操作提供了一个可以抛出错误的版本,tryMap。使用 tryMap 我们就可以将这类处理数据时发生的错误转变为标志事件流失败的结束事件。

["1", "2", "S", "4"].publisher
    .tryMap { s -> Int in
        guard let value = Int(s) else {
            throw MyError.myError
        }
        return value
    }

除了 tryMap 以外,Combine 中还有很多类似的以 try 开头的 Operator,比如 tryScan,tryFilter,tryReduce 等等。当你有需求在数据转换或者处理时,将事件流以错误进行终止,都可以使用对应操作的 try 版本来进行抛出,并在订阅者一侧接收到对应的错误事件。

从错误中恢复

如果我们想要在事件流以错误结束时被转为一个默认值的话,replaceError 就会很有用。

replaceError 会将 Publisher 的 Failure 类型抹为 Never,这正是我们使用 assign 来将 Publisher 绑定到 UI 上时所需要的 Failure 类型。我们可以用 replaceError 来提供这样一个在出现错误时应该显示的默认值。

有一些 Operator 是专门帮助事件流从错误中恢复的,最简单的是 replaceError,它会把错误替换成一个给定的值,并且立即发送 finished 事件

replaceError(中断信号)

["1", "2", "Swift", "4"].publisher
    .tryMap { s -> Int in
        guard let value = Int(s) else {
            throw MyError.myError
        }
        return value
    }
    .replaceError(with: -1)
// Output:
// 1 2 -1

catch(中断信号)

catch 则略有不同,它接受的是一个新的 Publisher,当上游 Publisher 发生错误时,catch 操作会使用新的 Publisher 来把原来的 Publisher 替换掉。

["1", "2", "Swift", "4"].publisher
    .tryMap { s -> Int in
        guard let value = Int(s) else {
            throw MyError.myError
        }
        return value
    }
    .catch { _ in [-1, -2, -3].publisher }

 // Output:
 // 1 2 -1 -2 -3
 
 // 当错误发生后,原本的 Publisher 事件流将被中断,取而代之,则是由 catch 所提供的事件流继续向后续的 Operator 及 Subscriber 发送事件。原来 Publisher 中的最后一个元素 “4”,将没有机会到达。

组合实现继续输入

如果我们将 (由 ["1", "2", "Swift", "4"] 构成的) 原 Publisher 看作是用户输入,将结果的 Int 看作是最后输出,那么像上面那样的方式使用 replaceError 或者 catch 的话,一旦用户输入了不能转为 Int 的非法值 (如 “Swift”),整个结果将永远停在我们给定的默认恢复值上,接下来的任意用户输入都将被完全忽略。这往往不是我们想要的结果,一般情况下,我们会想要后续的用户输入也能继续驱动输出,这时候我们可以靠组合一些 Operator 来完成所需的逻辑

check("Catch and Continue") {
    ["1", "2", "Swift", "4"].publisher
        .flatMap { s in
            return Just(s)
                .tryMap { s -> Int in
                    guard let value = Int(s) else {
                        throw MyError.myError
                    }
                    return value
                }
                .catch { _ in
                    Just(-1)}
        }
}

// Output:
// 1 2 -1 4

eraseToAnyPublisher

类型抹消

let a = [[1,2,3],[4,5,6]].publisher.flatMap{ $0.publisher }

// 等价于
let a1 = Publishers.FlatMap(upstream: [[1,2,3],[4,5,6]].publisher, maxPublishers: .unlimited) {
    $0.publisher
}

let b = a.map {$0*2}
// 此时 b 的类型为: Publishers.Map<Publishers.FlatMap<Publishers.Sequence<[Int], Never>, Publishers.Sequence<[[Int]], Never>>, Int>

Combine 提供了 eraseToAnyPublisher 来帮助我们对复杂类型的 Publisher 进行类型抹消,这个方法返回一个 AnyPublisher。

let c = b.eraseToAnyPublisher()
// c: AnyPublisher<Int, Never>

Combine 中的其他角色也大都提供了类似的抹消后的类型,比如 AnySubscriber 和 AnySubject 等。在大多数情况下我们都只会关注某个部件所扮演的角色,也即,它到底是一个 Publisher 还是一个 Subscriber。一般我们并不关心具体的类型,因为对 Publisher 的变形往往都伴随着类型的变化。通过类型抹消,可以让事件的传递和订阅操作变得更加简单,对外的 API 也更加稳定。

操作符熔合

将操作符的作用时机提前到创建 Publisher 时的方式,被称为操作符熔合 (operator fusion)。

有时候 map 操作的返回结果的类型并不是 Publishers.Map,比如下面的两个例子:

[1, 2, 3].publisher.map { $0 * 2 }
// Publishers.Sequence<[Int], Never>
Just(10).map { String($0) }
// Just<String>

这是由于 Publishers.Sequence 和 Just 在各自的扩展中对默认的 Publisher 的 map 操作进行了重写。由于 Publishers.Sequence 和 Just 这样的类型在编译期间我们就能确定它们在被订阅时就会同步地发送所有事件,所以可以将 map 的操作直接作用在输入上,而不需要等待每次事件发生时再去操作。

可以想象 Publishers.Sequence 上 map 操作的实现:

extension Publishers.Sequence {
public func map<T>(_ transform: (Elements.Element) -> T) -> Publishers.Sequence<[T], Failure> {
    return Publishers.Sequence(sequence: sequence.map(transform))
    }
}

这样做可以避免 transform 的 @escaping 的要求,避免存储这个变形闭包,让所得到的 Publisher 更加高效,同时也让后续 Publisher 变形类型能简单一些。这些优点在多次变形的时候尤为突出。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,718评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,683评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,207评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,755评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,862评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,050评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,136评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,882评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,330评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,651评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,789评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,477评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,135评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,864评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,099评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,598评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,697评论 2 351

推荐阅读更多精彩内容