强制发布者发布更新?

时间:2021-03-14 06:54:52

标签: swift swiftui combine

在我的 init()ViewModel 中,我有 2 个订阅者。

  1. 第一个订阅 DataProvider() 内下载的数据并使用结果更新 downloadedData。这会进行 API 调用,通常需要几秒钟才能发布数据。
  2. 第二个订阅 $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"])
            }
        }
    }

2 个答案:

答案 0 :(得分:2)

您可能以错误的方式思考问题。

首先,让我们正确理解术语。您正在订阅init 中的 2 个发布商。每当它们各自的属性发生变化时,这两个发布者都会发出一个值。

您似乎想要做的是在 filteredDatasavedData 发生变化时更新 downloadedData。因此,与其“强制更新”,不如订阅那种在需要时发出所需内容的发布者。您可以使用 CombineLatest 来实现:

$savedData.combineLatest(provider.$data)
   .map { (saved, downloaded) in
      //...
   }
   .sink { [weak self] value in
      self?.filteredData = value
   }
   .store(in: &cancellables)

(不要使用 assign - 它会创建对 self 的强引用;改为使用 sinkweak self

答案 1 :(得分:0)

您可以订阅downloadedData并将其分配给保存的数据:

$downloadedData
    .assign(to: \.savedData, on: self)
    .store(in: &cancellables)

请参考以下链接,当将 assign(to: on:) 作为参数传递时,使用 self 会导致内存泄漏。

https://symfony.com/doc/current/security/user_provider.html#comparing-users-manually-with-equatableinterface

您的其他订阅也使用 assign(to: on:) 作为参数调用 self,最好在链接中实施解决方案,因为它可以在整个应用程序中解决此问题。