在我的 init()
的 ViewModel
中,我有 2 个订阅者。
DataProvider()
内下载的数据并使用结果更新 downloadedData
。这会进行 API 调用,通常需要几秒钟才能发布数据。$savedData
,然后仅使用 filteredData
AND savedData
中包含的数据更新 downloadedData
数组.问题是 downloadedData
需要时间来下载,而 savedData
是即时的(它实际上在 Core Data 中)。下载后,我想强制更新上面的第二个观察者 ($savedData
) 以相应地更新 filteredData
。
有没有办法手动强制发布者发布一个值?在 ObservedObjects 上,我们可以调用 objectWillChange.send()
,但是有没有类似的东西我可以在 $savedData
上调用?
免责声明:这是我的应用程序的超级简化版本。请不要包含改变层次结构的答案。我也意识到我可以在 $allCoins 上添加另一个发布者来模仿 $savedData 上的发布者,但我宁愿只是强制更新 $savedData。谢谢!
import SwiftUI
struct TestView: View {
@StateObject var vm = ViewModel()
var body: some View {
VStack {
ForEach(vm.filteredData, id: \.self) { user in
Text(user)
}
}
}
}
struct TestView_Previews: PreviewProvider {
static var previews: some View {
TestView()
}
}
import Combine
class ViewModel: ObservableObject {
@Published var downloadedData: [String] = []
@Published var savedData: [String] = []
@Published var filteredData: [String] = []
var cancellables = Set<AnyCancellable>()
@Published var provider = DataProvider()
init() {
// WHEN THIS GETS PUBLISHED...
// CAN I FORCE A PUBLISH / UPDATE TO savedData?
provider.$data
.assign(to: \.downloadedData, on: self)
// here?
.store(in: &cancellables)
$savedData
.map { (saved) -> [String] in
return saved.map { (savedItem) -> String in
return self.downloadedData.contains(savedItem) ? savedItem : "NO"
}
}
.assign(to: \.filteredData, on: self)
.store(in: &cancellables)
getSavedData()
}
func getSavedData() {
savedData = ["ONE", "TWO"]
}
}
class DataProvider: ObservableObject {
@Published var data: [String] = []
init() {
getData()
}
func getData() {
// simulating api call
DispatchQueue.main.asyncAfter(deadline: .now() + 2.0) {
self.data.append(contentsOf: ["ONE", "TWO"])
}
}
}
答案 0 :(得分:2)
您可能以错误的方式思考问题。
首先,让我们正确理解术语。您正在订阅init
中的 2 个发布商。每当它们各自的属性发生变化时,这两个发布者都会发出一个值。
您似乎想要做的是在 filteredData
或 savedData
发生变化时更新 downloadedData
。因此,与其“强制更新”,不如订阅那种在需要时发出所需内容的发布者。您可以使用 CombineLatest
来实现:
$savedData.combineLatest(provider.$data)
.map { (saved, downloaded) in
//...
}
.sink { [weak self] value in
self?.filteredData = value
}
.store(in: &cancellables)
(不要使用 assign
- 它会创建对 self
的强引用;改为使用 sink
和 weak self
)
答案 1 :(得分:0)
您可以订阅downloadedData并将其分配给保存的数据:
$downloadedData
.assign(to: \.savedData, on: self)
.store(in: &cancellables)
请参考以下链接,当将 assign(to: on:)
作为参数传递时,使用 self
会导致内存泄漏。
您的其他订阅也使用 assign(to: on:)
作为参数调用 self
,最好在链接中实施解决方案,因为它可以在整个应用程序中解决此问题。