解决 Swift 6 全局变量不能满足并发安全(concurrency-safe)读写的问题

news2025/1/12 22:57:39

在这里插入图片描述

概述

WWDC 24 终于在 Swift 十岁生日发布了全新的 Swift 6。这不仅意味着 Swift 进入了全新的“大”版本时代,而且 Swift 编译器终于做到了并发代码执行的“绝对安全”。

在这里插入图片描述

不过,从 Swift 5 一步迈入“新时代”的小伙伴们可能对新的并发检查有些许“水土不服”,比如在同步全局变量时可能会引发 Swift 6 编译器的“怨声载道”。

在本篇博文中,您将学到如下内容:

  • 概述
  • 1. 由一个并发错误引发的“探险”
  • 2. 拯救者:全局 Actor!
  • 3. 另一种解决之道
  • 总结

驯服 Swift 6 并发代码执行中的安全隐患是每个秃头码农心中的“壮志凌云”,那还等什么呢?让我们马上将“野性难驯”的 Concurrency 执行驯化的俯首帖耳吧!

Let‘s go!!!😉


1. 由一个并发错误引发的“探险”

在将 Xcode 项目的编译器版本从 Swift 5 升级至 6 之后,小伙伴们会发之前很多“安适如常”的并发代码突然变得东差西误起来。
在这里插入图片描述

比如,下面一段简单的代码在 Swift 6 环境中运行就仿佛会“魂不守舍”一般:

var id = UUID()

func test() async {
    id = UUID()
}

除了全局变量以外,对于结构或类的静态变量也会有类似的问题:

struct Model {
    static var id = UUID()
    
    func test() async {
        id = UUID()
    }
}

在这里插入图片描述

这都是怎么回事呢?

在这里插入图片描述

原来,从 Swift 6 开始苹果对并发执行安全性的态度变得极度较真起来。为了不惯着 Swift 5 中模棱两可的并发错误,苹果采取“零容忍”策略让代码中所有潜在的并发陷阱都变得“原形毕露”了!

在这里插入图片描述

在上面的代码中,我们在异步 test 方法中修改了全局变量,而该方法有可能会被并发执行。这绝对是一个非常严重的并发同步错误。除了将全局变量 id 变为只读(在这个例子中显然不可行)以外,还有其它解决良策么?

答案是:必须的!

2. 拯救者:全局 Actor!

为了让全局变量不受并发执行同步的困扰之苦,我们可以简单的将它们用 MainActor 来修饰:

@MainActor
var id = UUID()

@MainActor
func test() async {
    id = UUID()
}

不过这样一来,我们就必须把所有涉及全局变量的读写操作都限制在主线程上,这无疑加重了主线程中不必要的执行负担。

然而,我们可以创建自己的全局 Actor ,然后用这些自定义全局 Actor 来让之前的同步问题迎刃而解:

@globalActor
actor MyActor {
    static let shared = MyActor()
}

@MyActor
var id = UUID()

@MyActor
func test() async {
    id = UUID()
}

@MyActor
struct Model {
    static var id = UUID()
    
    func test() async {
        Self.id = UUID()
    }
}

或者不用全局 Actor 来修饰 test 方法,而在 test 内部做文章:

func test() async {
    Task {@MyActor in
        id = UUID()
    }
}

这样一来,我们就可以将所有对全局变量的读写限制在特定的 Actor 之中,从而大功告成!


更多关于 Swift 语言中 Actor 的更多介绍,请小伙伴们移步如下链接进一步观赏精彩内容:

  • Swift 结构化并发之全局 Actor 趣谈
  • 深入理解 Swift 新并发模型中 Actor 的重入(Reentrancy)问题
  • SwiftUI async/await 并发代码提示 Non-sendable type cannot cross actor boundary 警告的解决

3. 另一种解决之道

除了用上面的方法以外,如果可以确定对全局变量的并发读写绝对不会引发同步问题的话,我们还可以显式让编译器“明明白白我们的心”:

nonisolated(unsafe)
var id = UUID()

func test() async {
    id = UUID()
}

如上代码所示:我们用 nonisolated(unsafe) 修饰语句明确告知 Swift 6 编译器,id 全局变量绝对不会导致并发执行中的同步问题。但这样一来,编译器将会对该全局变量在同步执行中所有的并发潜在问题置之不理、一笑而过。所以,这等于将全部的并发安全检查又重新转移到了小伙伴们自身的肩头。

使用 nonisolated(unsafe) 修饰语句还有一种可能是:明知这些全局变量可能发生并发同步问题,但暂时先将他们搁置一边,等以后再集中火力攻克它们。

希望大家在使用这种方法时慎之又慎,切记切记!


想要系统学习 Swift 的小伙伴们,可以移步我的《Swift 语言开发精讲》专栏来看一看哦:

在这里插入图片描述

  • Swift 语言开发精讲

总结

在本篇博文中,我们讨论了在 Swift 6 中异步并发读写全局变量时让编译器“大发雷霆”的原因,并给出多种解决之道,棒棒哒!

感谢观赏,再会啦!😎

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2053078.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

连锁美业门店收银系统拓客系统预约系统Java源码-博弈美业APP如何进行课程核销?

* 课程开课后,到课程结束前,这段时间均可以进行课程核销 * 课程核销的权限,仅限内部员工 * 核销课程时,需要切换到总部,才有核销课程的权限 方法一:通过“课程核销”直接核销 点击“课程核销”&#xff…

强大的接口测试可视化工具:Postman Flows

Postman Flows是一种接口测试可视化工具,可以使用流的形式在Postman工作台将请求接口、数据处理和创建实际流程整合到一起。如下图所示 Postman Flows是以API为中心的可视化应用程序开发界面。它提供了一个无限的画布用于编排和串连API,数据可视化来显示…

[C#]winform基于深度学习算法MVANet部署高精度二分类图像分割onnx模型高精度图像二值化

【训练源码】 https://github.com/qianyu-dlut/MVANet 【参考源码】 https://github.com/hpc203/MVANet-onnxrun 【算法介绍】 二分图像分割(DIS)最近出现在从高分辨率自然图像中进行高精度对象分割方面。在设计有效的DIS模型时,主要的挑战是…

XMGoat:一款针对Azure的环境安全检测工具

关于XMGoat XMGoat是一款针对Azure的环境安全检测工具,XM Goat 由 XM Cyber Terraform 模板组成,可帮助您了解常见的 Azure 安全问题。每个模板都是一个用于安全技术学习的靶机环境,包含了一些严重的配置错误。 在该工具的帮助下&#xff0c…

景区门票预订系统开发方案概述

随着旅游业的蓬勃发展,提升游客体验、优化景区管理成为了各大景区亟待解决的问题。景区门票预订系统的开发,正是顺应这一趋势的重要举措。以下是一个简要的景区门票预订系统开发方案概述,旨在通过科技手段实现票务管理的智能化、便捷化。 一、…

faiss向量数据库测试《三体》全集,这家国产AI加速卡,把性能提了7倍!

在人工智能和机器学习技术的飞速发展中,向量数据库在处理高维数据方面扮演着日益重要的角色。近年来,随着大型模型的流行,向量数据库技术也得到了进一步的发展和完善。 向量数据库为大型模型提供了一个高效的数据管理和检索平台,…

如何将音乐保存为文件格式为铃声,怎么把音乐保存为MP3格式

许多小伙伴在听到好听的音乐时,会将其下载保存。如果需要将音乐文件格式转换成来电铃声时,就需要借助专业的音频处理软件了,音频处理软件可以帮助我们转化音乐文件格式,那么下面就来给大家介绍如何将音乐保存为文件格式为铃声&…

Kali Linux-设置系统24小时时间制

文章目录 设置系统24小时时间制 设置系统24小时时间制 在Kali Linux中,如果系统时间不是以24小时制显示,你可以通过修改系统时间格式配置文件来调整为24小时制。以下是具体的操作步骤: 1.编辑/etc/locale.conf配置文件。 vim /etc/locale.c…

【gitlab】gitlab-ce:17.3.0-ce.0 之2:配置

参考阿里云的教程docker的重启 sudo systemctl daemon-reload sudo systemctl restart docker配置 –publish 8443:443 --publish 8084:80 --publish 22:22 sudo docker ps -a 當容器狀態為healthy時,說明GitLab容器已經正常啟動。 root@k8s-master-pfsrv:~

Python WebSocket自动化测试:构建高效接口测试框架

为了更高效地进行WebSocket接口的自动化测试,我们可以搭建一个专门的测试框架。本文将介绍如何使用Python构建一个高效的WebSocket接口测试框架,并重点关注以下四个方面的内容:运行测试文件封装、报告和日志的封装、数据驱动测试以及测试用例…

调查显示:超过30 %企业遭受过网络攻击,如何防范?

对于企业来讲,屡禁不止的网络安全攻击始终是阻碍业务发展的重大隐患。调查结果显示,有近8成的企业将网络安全视为企业很重要的工作之一,另外,有超过三成的企业遭受过网络安全攻击。 企业作为网络安全事故的责任承担方,…

synchronized和Lock(ReentrantLock)及二者区别

synchronized 是用于实现线程同步的关键字。它提供了两种主要的方式来保证多个线程访问共享资源时的互斥性和可见性:同步块和同步方法。 同步块 同步块允许你指定一个对象作为锁,并保护一段代码区域。这样,同一时刻只有一个线程可以执行这…

ChatGPT 为什么不建议关闭 MySQL 严格模式?

社区王牌专栏《一问一实验:AI 版》全新改版归来,得到了新老读者们的关注。其中不乏对 ChatDBA 感兴趣的读者前来咨询,表达了想试用体验 ChatDBA 的意愿,对此我们表示感谢 🤟。 目前,ChatDBA 还在最后的准备…

记录一次 npm ERR! cb() never called! 解决过程

gitlab cicd过程,使用docker部署Vue3前端项目,报错如下: 针对 npm ERR! cb() never called! 这个报错,网上有很多解决方案,大都是清空缓存,重新运行npm 之类的。笔者全都试过,无法解决问题。笔者…

linux,docker查看资源消耗总结

在linux和docker中我们将一个程序运行到后台,之后我们想查看它的运行状态,对于服务器的资源消耗等等 1.linux查看进程 ps aux | grep python ps aux:列出所有正在运行的进程。grep python:过滤出包含 python 的进程 2.linux查…

springCloud集成activiti5.22.0流程引擎(分支)

springCloud集成activiti5.22.0流程引擎 点关注不迷路,欢迎再访! 精简博客内容,尽量已行业术语来分享。 努力做到对每一位认可自己的读者负责。 帮助别人的同时更是丰富自己的良机。 文章目录 springCloud集成activiti5.22.0流程引擎一.Sprin…

你知道家电的保质期吗?

家人们,你们有关注过家里的电器用了多少年了吗? “家电不坏,就能一直用。” “坏了修一修,一样能用。” 很多家长都有这样的想法,家里的电器即使出了故障,修一修也就继续用了。 其实,家电也…

打造智能化直播商城平台:AI与大数据在平台开发中的应用

在当今竞争激烈的电商市场中,直播商城平台已经成为品牌和商家实现差异化竞争的重要工具。随着人工智能(AI)和大数据技术的不断进步,智能化直播商城平台的开发成为了行业的新趋势。这些技术不仅可以优化用户体验,还能提…

AI革新体育:IBM携手USTA升级美国公开赛观赛体验

IBM和美国网球协会(USTA)合作,在2024年美国网球公开赛中引入了创新的AI技术,为观众和选手带来全新的体验。8月19日赛事开幕,IBM的watsonx平台将推出多项新功能,增强三周赛事的互动性。 喜好儿网 IBM的Gra…

Prometheus:pushgateway使用

1 项目目标 (1)熟练部署pushgateway (2)使用api增删改查数据 (3)使用python Client SDK Push数据到pushgateway 2.1 规划节点 主机名 主机IP 节点规划 prome-master01 10.0.1.10 服务端 prome-no…