I was experimenting with writing Global Actor code on hand while watching SE-0316. At that time, even though @MainActor was given, it was not executed on the main thread. Here is the code:
Adding async to the closure's type annotation executed it in the main thread as intended. Is the earlier code a bug? Please let me know if there is any knowledge that is missing, not a bug.
In that example, it would be true. It's ok to understand.
Detatched Task does not inherit actor context, but since UIViewContrller is implicitly isolated to MainActor, the callback is isolated to MainActor, so it is executed on the main thread.
You could also explicitly call over to the MainActor if thatâs easier to read:
Task {
MainActor.run {
await self.callback()
}
}
I sometimes find this easier for me to remember what I was aiming for. Kind of the equivalent to âmain.async { }â from the DispatchQueue technique.
I checked additionally, and when I removed the type annotation as shown below, it was executed on the main thread. It seems that the type annotation I wrote was incorrect.
import UIKit
class ViewController: UIViewController {
var callback = { @MainActor in
print(Thread.isMainThread) // true
}
override func viewDidLoad() {
super.viewDidLoad()
Task.detached {
await self.callback()
}
}
}
Hmm, perhaps the type of callback is not "() -> Void" but "() async -> Void" in this case, as ViewController is implicitly an actor by virtue of UIViewController?
Right. Actor isolation has to be represented in the type of a sync function, or else it just canât be automatically honored â itâs sync, so only the caller can make sure it runs with the right isolation. In non-strict mode, Swift has to assume that youâre getting it right dynamically somehow, but strict mode makes the compiler diagnose it immediately. SE-0423, if accepted, will at least make Swift diagnose the failure dynamically.
Sorry for jumping with my topic but I have a slightly different yet similar problem with understanding how @MainActor annotation is preserved while passing around a closure.
Could someone look at why it is not respected even if the closure type is annotated with @MainActor?