异步等待调用后更新 UI

Iva*_*lov 11 async-await swift xcode13

我从 API 加载书籍,加载时显示活动指示器,在服务器响应后更新标签。

activityView.isHidden = false
        
let task = detach {
    do {
        let books = try await self.bookService.fetchBooks()
        DispatchQueue.main.async {
            self.show(books: books)
        }
    } catch {
        DispatchQueue.main.async {
            self.resultLabel.text = error.localizedDescription
        }
    }
    DispatchQueue.main.async {
       self.activityView.isHidden = true
    }
}

//...
Run Code Online (Sandbox Code Playgroud)

我的问题是更新主队列上的 UI 的更好方法是什么?DispatchQueue.main.async看起来很丑,我想有更好的方法来做同样的事情。

我必须使用它,因为所有 UI 更新都应该在主线程上,并且我会收到编译器错误,而没有DispatchQueue.main.async类似的内容

与全局参与者“MainActor”隔离的属性“文本”无法从非隔离上下文中发生突变

或者

与全局参与者“MainActor”隔离的属性“isHidden”无法从非隔离上下文中进行突变

PS使用Xcode 13.0b2

Tar*_*agi 11

@MainActor像这样使用-

self.updateAcitivityIndicator(isHidden: false)
        
let task = detach {
    do {
        let books = try await self.bookService.fetchBooks()
        self.showBooks(books)
    } catch {
        self.showError(error)
    }
    self.updateAcitivityIndicator(isHidden: true)
}

@MainActor
private func showBooks(_ books: [Book]) {
}

@MainActor
private func showError(_ error: Error) {
    self.resultLabel.text = error.localizedDescription
}

@MainActor
private func updateAcitivityIndicator(isHidden: Bool) {
    self.activityView.isHidden = isHidden
}
Run Code Online (Sandbox Code Playgroud)

  • @jatinfl 这些是不同的概念,“@MainActor”是与“async”/“await”一起引入的,正是为了这个目的OP需要解决。请观看与 Swift Concurrency 相关的 **WWDC 2021** 会议,您就会知道您需要什么。 (2认同)