在主线程上使用@Published值吗?

时间:2019-11-26 22:39:31

标签: swiftui combine

是否有一种方法可以指定计数仅应在主线程上发布?我见过一些有关使用“ 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
}

2 个答案:

答案 0 :(得分:1)

如果您尝试从后台线程设置标记为@Published的值,则会看到此错误:

  

不允许从后台线程发布更改;确保   从主线程发布值(通过诸如receive

之类的运算符

因此,您必须确保在将值设置为在主线程上完成此操作的任何地方,这些值将始终发布在主线程上。

答案 1 :(得分:1)

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

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

还可以查看heckj's examples,我发现它们非常有用。