Skip to content

Commit

Permalink
Correcting inaccurate information regarding function execution (#443)
Browse files Browse the repository at this point in the history
* Inaccurate last section regarding Task inheritance replaced with a new one documenting the correct assumption regarding async function execution.

* Update _posts/2023-11-14-swift-concurrency-in-a-nutshell.md

PR feedback

Co-authored-by: Sébastien Drode <[email protected]>

* New details about task in SwiftUI

---------

Co-authored-by: Sébastien Drode <[email protected]>
  • Loading branch information
DamienPetrilli and Dragna authored Apr 8, 2024
1 parent 26ddbf1 commit d5a3ffe
Showing 1 changed file with 19 additions and 19 deletions.
38 changes: 19 additions & 19 deletions _posts/2023-11-14-swift-concurrency-in-a-nutshell.md
Original file line number Diff line number Diff line change
Expand Up @@ -500,32 +500,29 @@ actor Counter {

In this example, while `process()` is awaiting the completion of `doLongProcessing()`, there's an opportunity for another task to call `increment()`. This undermines the expectation that an actor's state remains consistent within a given method. So, the second `print(value)` may output an unpredictable result, illustrating the challenge of managing mutable state in an actor with reentrant behavior.

### Unintentional Task Inheritance
### Async Function Execution Contexts

In Swift's concurrency model, child tasks inherit the properties of their parent tasks by default, including priority levels and task-local values. Lack of awareness about this behavior can lead to unexpected outcomes, particularly when generating Tasks through SwiftUI modifiers, as demonstrated below.
Contrary to the behavior in Grand Central Dispatch (GCD), where all code executed within the scope of a block is performed on the same thread, Swift's concurrency model executes any `async` function on a global executor unless explicitly specified otherwise, such as with the `@MainActor` annotation.

Note: a `.task{}` in SwiftUI [runs implicitely](https://oleb.net/2022/swiftui-task-mainactor/) on the MainActor when set within the `body` of a SwiftUI View.

```swift
struct MyView: View {
var body: some View {
...
}
var body: some View {
...
.task {
await fetchData()
// Code within this block is executed on the Main Actor.
print("hello")
// Executed on a Global Executor despite being called from the Main Actor.
await fetchData()
// Executed on the Main Actor because we explicitly used @MainActor below.
await updateUI()
}

func fetchData() async {
Task {
// Inherits properties (e.g., priority, executor) from the parent Task
// The long job will execute on the main thread
await longJob()
}
}

Task.detached(priority: .userInitiated) {
// Unstructured Task: Does not inherit any properties from parent Task
// The long job will execute outside the main thread
await longJob()
}
}

func fetchData() async { ... }
@MainActor func updateUI() async { ... }
}
```

Expand All @@ -543,3 +540,6 @@ As we have seen, Swift Concurrency is a huge step forward in terms of safety and
- [Actors](https://github.com/apple/swift-evolution/blob/main/proposals/0306-actors.md)
- [Global Actors](https://github.com/apple/swift-evolution/blob/main/proposals/0316-global-actors.md)
- [Concurrency is not Parallelism](https://youtu.be/oV9rvDllKEg?si=kwXQULVlNNT3K6LS)
- [How to determine where code runs in Swift Concurrency](https://www.youtube.com/watch?v=8T4XuCM0abI)
- [Your Brain 🧠 on Swift Concurrency - iOS Conf SG 2023](https://www.youtube.com/watch?v=zgCtube1DSg)
- [Where View.task gets its main-actor isolation from](https://oleb.net/2022/swiftui-task-mainactor/)

0 comments on commit d5a3ffe

Please sign in to comment.