简体   繁体   中英

How can I use async/await with SwiftUI in Swift 5.5?

I have been testing the async/await functionality previewed in the Swift 5.5 release, but I am unable to collect the results from an async function and display them using SwiftUI. Here is my code:

import SwiftUI

struct AsyncTestView: View {
    @State var text: String?

    // Async function
    func asyncGetText() async -> String {
        Thread.sleep(forTimeInterval: 10)
        return "My text"
    }
    
    // Stores the result of async function
    func fetchText() async {
        let text = await asyncGetText()
        DispatchQueue.main.async {
            self.text = text
        }
    }
    
    var body: some View {
        Text(text ?? "Loading")
            .onAppear(perform: fetchText)
    }
}

This results in the following error:

'async' call in a function that does not support concurrency
Add 'async' to function 'fetchText()' to make it asynchronous

Adding async to the fetchText() function then results in the following error on the .onAppear() function:

Invalid conversion from 'async' function of type '() async -> ()' to synchronous function type '() -> Void'

In this article , they use the @asyncHandler tag to annotate the fetchText function, however this results in the warning: '@asyncHandler' has been removed from the language' .

I'm the author of the article you referenced.

As discussed in Discover concurrency in SwiftUI , views can make use of the new .task { } and .refreshable { } modifiers to fetch data asynchronously.

So you now have the following options to call you async code:

func someSyncMethod() {
  doSomeSyncWork()
  async {
    await methodThatIsAsync()
  }
}
List {
}
.task {
  await methodThatIsAsync()
}
List {
}
.refreshable {
  await methodThatIsAsync()
}

If you're using a separate view model, make sure to mark it as @MainActor to ensure property updates get executed on the main actor.

I updated the code for my article: https://github.com/peterfriese/Swift-Async-Await-Experiments

As per new informations in WWDC session Meet async/await in Swift WWDC , at 23m:28s this is now done using:

async {
   await someAsyncFunction()
}

Screenshot from the session.

在此处输入图像描述

I agree with @peter-friese's answer but will add for those reading this the change of syntax when bridging from sync to async:

The new syntax:

Task {
   await someAsyncFunction()
}

Replaces this syntax:

async {
   await someAsyncFunction()
}

... and the Task() initialiser can accept a priority parameter:

Task(priority: .userInitiated) {
   await someAsyncFunction()
}
import SwiftUI

struct AsyncTestView: View {
    @State var text = “Loading”

    // Async function
    func asyncGetText() async -> String {
        Thread.sleep(forTimeInterval: 10) // Maybe should be Task.sleep
        return "My text"
    }
    
    var body: some View {
        Text(text)
        .task {
             text = await asyncGetText()
        }
    }
}

The technical post webpages of this site follow the CC BY-SA 4.0 protocol. If you need to reprint, please indicate the site URL or the original address.Any question please contact:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM