Swift 组合运算符与 RxSwift 框架中的“withLatestFrom”等功能相同

Ali*_*ash 5 ios swift swiftui combine

我正在开发一个采用 MVVM 模式的 iOS 应用程序,使用 SwiftUI 设计视图和 Swift 组合,以便将我的视图与其各自的视图模型粘合在一起。在我的ViewModels之一,我已经创建了一个Publisher(类型Void为按下一个按钮,另一个用于A的含量)TextField(类型String)。我希望能够在我的 ViewModel 中组合两个发布者,这样组合的发布者仅在按钮发布者发出事件时才发出事件,同时从字符串发布者获取最新事件,因此我可以对TextField数据进行某种评估, 每次用户按下按钮时。所以我的虚拟机看起来像这样:

import Combine
import Foundation

public class MyViewModel: ObservableObject {
    @Published var textFieldContent: String? = nil
    @Published var buttonPressed: ()

    init() {
        // Combine `$textFieldContent` and `$buttonPressed` for evaulation of textFieldContent upon every button press... 
    }
}
Run Code Online (Sandbox Code Playgroud)

两个发布者都通过 SwiftUI 获取数据,所以我将省略那部分,让我们假设两个发布者都随着时间的推移收到一些数据。

来自 RxSwift 框架,我的 goto 解决方案是使用withLatestFrom运算符来组合两个可观察对象。然而,在“组合来自多个出版商的元素”一节中深入研究出版商的 Apple 文档,我找不到类似的东西,所以我希望目前缺少这种运算符。

所以我的问题是:是否有可能使用结合框架的现有运算符 API 来最终获得相同的行为withLatestFrom

mat*_*att 8

有一个内置的运算符听起来很棒,但是您可以从现有的运算符中构建相同的行为,如果这是您经常做的事情,那么很容易从现有的运算符中创建自定义运算符。

这种情况下的想法是combineLatest与运算符一起使用,例如removeDuplicates,除非按钮发出新值,否则防止值沿管道传递。例如(这只是在操场上的一个测试):

var storage = Set<AnyCancellable>()
var button = PassthroughSubject<Void, Never>()
func pressTheButton() { button.send() }
var text = PassthroughSubject<String, Never>()
var textValue = ""
let letters = (97...122).map({String(UnicodeScalar($0))})
func typeSomeText() { textValue += letters.randomElement()!; text.send(textValue)}

button.map {_ in Date()}.combineLatest(text)
    .removeDuplicates {
        $0.0 == $1.0
    }
    .map {$0.1}
    .sink { print($0)}.store(in:&storage)

typeSomeText()
typeSomeText()
typeSomeText()
pressTheButton()
typeSomeText()
typeSomeText()
pressTheButton()
Run Code Online (Sandbox Code Playgroud)

输出是两个随机字符串,例如"zed""zedaf"。关键是每次我们调用 时typeSomeText,都会将文本发送到管道中,但是除非我们调用 ,否则我们不会在管道末端接收到文本pressTheButton

这似乎是你所追求的那种事情。


你会注意到我完全忽略了按钮发送的值是什么。(在我的示例中,无论如何它只是一个空值。)如果该值很重要,则更改初始映射以将该值作为元组的一部分包含在内,然后删除元组的 Date 部分:

button.map {value in (value:value, date:Date())}.combineLatest(text)
    .removeDuplicates {
        $0.0.date == $1.0.date
    }
    .map {($0.value, $1)}
    .map {$0.1}
    .sink { print($0)}.store(in:&storage)
Run Code Online (Sandbox Code Playgroud)

这里的重点是,在该行之后到达的.map {($0.value, $1)}内容与withLatestFrom将产生的内容完全相同:两个发布者的最新值的元组。

  • 这不是 excat `withLatestFrom` https://rxjs.dev/api/operators/withLatestFrom。WithLatestFrom 应该“仅当源发出时”发出事件 ---- ``` pressTheButton() typeSomeText() ``` 此代码打印字符,但不应打印字符 (3认同)