首页 > 解决方案 > 在主线程上使用 @Published 值?

问题描述

有没有办法指定 count 应该只在主线程上发布?我看过有关使用 Publisher 设置您的文档的文档receive(on:),但在这种情况下,@Publisher包装器隐藏了该逻辑。

import SwiftUI
import Combine

class MyCounter: ObservableObject {
  @Published var count = 0

  public static let shared = MyCounter()
  
  private init() { }
}

struct ContentView: View {
    @ObservedObject var state = MyCounter.shared
    var body: some View {
        return VStack {
            Text("Current count: \(state.count)")
            Button(action: increment) {
                HStack(alignment: .center) {
                    Text("Increment")
                        .foregroundColor(Color.white)
                        .bold()
                }
            }
        }
    }
    
    private func increment() {
        NetworkUtils.count()
    }
}

public class NetworkUtils {

    public static func count() {
        guard let url = URL.parse("https://www.example.com/counter") else {
            return
        }
        
        var request = URLRequest(url: url)
        request.httpMethod = "GET"
        
        let task = URLSession.shared.dataTask(with: request) { data, response, error in
            if let response = response as? HTTPURLResponse {
                let statusCode = response.statusCode
                if statusCode >= 200 && statusCode < 300 {
                    do {
                        guard let responseData = data else {
                            return
                        }
                        
                        guard let json = try JSONSerialization.jsonObject(with: responseData, options: []) as? [String: Any] else {
                            return
                        }
                        
                        if let newCount = json["new_count"] as? Int{
                            MyCounter.shared.count = newCount
                        }
                    } catch  {
                        print("Caught error")
                    }
                }
            } 
        }
        task.resume()
    }   
}

从我更新的示例中可以看出,这是一个简单的 SwiftUI 视图,它有一个按钮,单击该按钮会进行网络调用。网络调用是异步的。当网络调用返回时,在ObservableObject MyCounter后台线程上更新。我想知道是否有办法ObservableObject在主线程上发布更改。我现在知道的唯一方法是将更新逻辑包装在网络调用闭包中,如下所示:

DispatchQueue.main.async {
    MyCounter.shared.count = newCount
}

标签: swiftswiftuicombine

解决方案


URLSession.shared.dataTask(with: request)您可以使用URLSession.shared.dataTaskPublisher(for: request)( docs ) 而不是 using ,这将允许您创建一个组合管道。然后,您可以将receive(on:)运算符链接为管道的一部分。

URLSession.shared.dataTaskPublisher(for: request)
  .map { response in ... }
  ...
  .receive(on: RunLoop.main)
  ...

另请查看heckj 的示例,我发现它们非常有用。


推荐阅读