Improve your app's performance by refactoring your code to take advantage of asynchronous functions in Swift.
- Note: This sample code project is associated with WWDC21 session 10194: Swift Concurrency: Update a Sample App.
Swift concurrency provides a standard set of language tools and techniques for concurrent programming. However, you may already have an existing project built with concurrency that uses other frameworks and techniques. You don't have to convert all of your code all at once; instead, you can use specific refactoring techniques to convert your code one piece at time.
This sample provides two separate versions of the Coffee Tracker app:
-
The original version uses completion handlers to query the HealthKit SDK and to respond to
CLKComplicationDataSource
calls, and dispatch queues to isolate concurrent access to memory. For more information on the original version, see Creating and Updating a Complication’s Timeline. -
The updated version uses Swift's concurrency features to provide clearer code with better error checking at compile time. It replaces the completion handlers with
async
functions, and uses actors to guarantee safe access to data.
Watch the session to see the process step by step, and then compare the two projects to see the differences.
To add the complication to an active watch face, start by building and running the sample code project in the simulator, and follow these steps:
- Click the Digital Crown to exit the app and return to the watch face.
- Using the trackpad, firmly press the watch face to put the face in edit mode, then tap Customize.
- Swipe left until the configuration screen highlights the complications. Select the complication to modify.
- Scroll to the Coffee Tracker complication, and then click the Digital Crown again to save your changes.
- Tap the Coffee Tracker complication to go back to the app.
For more information on setting up watch faces, see Change the watch face on your Apple Watch.
After configuring and running the Coffee Tracker app, you can test the background updates. Make sure the Coffee Tracker complication appears on the active watch face. Then build and run the app in Simulator, and follow these steps:
- Add one or more drinks using the app's main view.
- Click the Digital Crown to send the app to the background.
- Open Settings, and scroll down to Health > Health Data > Nutrition > Caffeine to see all of the drinks you added to the app.
- Click Delete Caffeine Data to clear all of the caffeine samples from HealthKit.
- Navigate back to the watch face.
Coffee Tracker updates the complication within 15 minutes; however, the update may be delayed based on the system's current state.
The HealthKitController
type contains several calls to the HealthKit SDK. In SDKs that support Swift concurrency, frameworks add async
-await
versions of most functions that previously took completion handlers.
You can remove completion handlers by updating these calls to use the async
-await
versions.
You suspend the store.save()
operation by adding the await
keyword.
Execution resumes after the await
completes.
An async
function can also be a throwing function, which you call by prepending try await
to the function call. Wrap the call in a do-catch
statement instead of using an Error?
type as a parameter to the completion handler.
// Save the sample to the HealthKit store.
do {
try await store.save(caffeineSample)
self.logger.debug("\(mgCaffeine) mg Drink saved to HealthKit")
} catch {
self.logger.error("Unable to save \(caffeineSample) to the HealthKit store: \(error.localizedDescription)")
}
In some cases an SDK call requires using a completion handler.
For example, a call to HKAnchoredObjectQuery.init(type:predicate:anchor:limit:resultsHandler:)
takes a completion handler, but the call that needs to await
is the call to HKHealthStore.execute(_:)
.
To await
the results of a completion handler in these cases, add a continuation
:
private func queryHealthKit() async throws -> ([HKSample]?, [HKDeletedObject]?, HKQueryAnchor?) {
return try await withCheckedThrowingContinuation { continuation in
// Create a predicate that only returns samples created within the last 24 hours.
let endDate = Date()
let startDate = endDate.addingTimeInterval(-24.0 * 60.0 * 60.0)
let datePredicate = HKQuery.predicateForSamples(withStart: startDate, end: endDate, options: [.strictStartDate, .strictEndDate])
// Create the query.
let query = HKAnchoredObjectQuery(
type: caffeineType,
predicate: datePredicate,
anchor: anchor,
limit: HKObjectQueryNoLimit) { (_, samples, deletedSamples, newAnchor, error) in
// When the query ends, check for errors.
if let error = error {
continuation.resume(throwing: error)
} else {
continuation.resume(returning: (samples, deletedSamples, newAnchor))
}
}
store.execute(query)
}
}
To protect the stored properties on the controller when accessed asynchronously, change HealthKitController
from a class
type to an actor
:
actor HealthKitController {
Calls to async
functions from synchronous functions are made by creating new asynchronous tasks, which can use await
to wait for completion:
// Handle background refresh tasks.
case let backgroundTask as WKApplicationRefreshBackgroundTask:
async {
// Check for updates from HealthKit.
let model = CoffeeData.shared
let success = await model.healthKitController.loadNewDataFromHealthKit()
if success {
// Schedule the next background update.
scheduleBackgroundRefreshTasks()
self.logger.debug("Background Task Completed Successfully!")
}
// Mark the task as ended, and request an updated snapshot, if necessary.
backgroundTask.setTaskCompletedWithSnapshot(success)
}
The CoffeeData
class implements ObservableObject
and has an @Published
property to feed the SwiftUI views.
To ensure that all updates to this property are made on the main thread, place the type on the main actor:
@MainActor
class CoffeeData: ObservableObject {
Two methods that perform synchronous IO --- the load
and save
methods --- are factored out into a separate CoffeeDataStore
actor, which performs these activities away from the main thread.
The model type on the main actor must use await
to call methods on the CoffeeDataStore
actor, which allows other work to run on the main thread during the synchronous IO operations.
The two types communicate by passing an array of Drink
values, which is a value type because Drink
is a structure. Loading returns an array of drinks, and saving takes an array of drinks as an argument.
To perform all methods asynchronously, replace the currentDrinks
property's didSet
operation with private(set)
and add a new async
method named drinksUpdated
. Move the code from the setter into the new method.
Call the drinksUpdated
after any code that sets the currentDrinks
property, using an await
call.
Update the drinksUpdated()
method to call the CoffeeDataStore
actor using an await
call.
The CoffeeDataStore
actor saves the data on a background thread.
Calls to the CoffeeData
object from SwiftUI views don't require any use of await
as these views are also on the main actor due to their use of @EnvironmentObject
.
Several methods on the CLKComplicationDataSource
protocol used to configure the app's timeline take completion handlers, which you can replace with their async
equivalents:
// Define whether the complication is visible when the watch is unlocked.
func privacyBehavior(for complication: CLKComplication) async -> CLKComplicationPrivacyBehavior {
// This is potentially sensitive data. Hide it on the lock screen.
.hideOnLockScreen
}