DispatchQueue.main.async

在Swift中编写异步代码时,你可能会遇到一些并发性和线程安全性的问题

例如刷新UI需要在主线程,在Swift 5.5之前我们都是使用手动来刷新主线程,这样既麻烦又容易遗漏

DispatchQueue.global().async {
    // 在后台线程执行耗时任务

    DispatchQueue.main.async {
        // 在主线程上更新用户界面
    }
}

@MainActor

Swift 5.5 引入了一个新的属性包装器 @MainActor,它提供了一种简单而安全的方式来在主线程上执行代码。

@MainActor 的原理是,它将属性或方法标记为只能在主队列(主线程)上执行。这意味着使用 @MainActor 修饰的代码块只能在主线程上运行,任何试图在其他线程上调用该代码块的尝试都将被阻塞,直到在主线程上运行。

使用 @MainActor 的语法很简单。你可以将它应用于属性或方法,示例代码如下:

class MyViewController: UIViewController {
    @MainActor
    var data: [String] = []

    @MainActor
    func updateUI() {
        // 更新用户界面的代码
    }
}

在上面的代码中,data 属性和 updateUI() 方法都被标记为 @MainActor。这意味着任何试图在非主线程上修改 data 属性或调用 updateUI() 方法的尝试都将被阻塞,直到在主线程上执行。

@MainActor 还可以应用于函数的参数,用于指示该参数必须在主线程上调用。例如:

@MainActor
func processUserInput(@MainActor completion: () -> Void) {
    // 处理用户输入的代码

    completion() // 在主线程上调用完成闭包
}

在上面的例子中,completion 参数被标记为 @MainActor,这意味着传递给该函数的闭包必须在主线程上调用。

@MainActor标记过的代码,只会在主线程中执行,所以就不会有concurrency(并发)的问题了

通过使用 @MainActor,你可以更轻松地确保你的代码在正确的线程上执行

在UIKit中使用@MainActor

现在我们看UIKit的API,都已经用@MainActor标记了。

当我们在使用UIKit时,即便不用DispatchQueue.main.async,也能确保在主线程上执行。

@MainActor open class UIButton : UIControl, NSCoding {} 

@MainActor open class UILabel : UIView, NSCoding, UIContentSizeCategoryAdjusting {}

@MainActor open class UIView : UIResponder, NSCoding, UIAppearance, UIAppearanceContainer, UIDynamicItem, UITraitEnvironment, UICoordinateSpace, UIFocusItem, UIFocusItemContainer, CALayerDelegate {}

// more

 在我们封装UIKit自定义控件时,为了保证线程安全,也需要@MainActor装饰

Logo

开放原子开发者工作坊旨在鼓励更多人参与开源活动,与志同道合的开发者们相互交流开发经验、分享开发心得、获取前沿技术趋势。工作坊有多种形式的开发者活动,如meetup、训练营等,主打技术交流,干货满满,真诚地邀请各位开发者共同参与!

更多推荐