Swift 异步序列 AsyncStream 新“玩法”以及内存泄漏、死循环那些事儿(上)

news2025/1/12 16:03:51

在这里插入图片描述

概览

异步序列(Async Sequence)是 Swift 5.5 新并发模型中的一员“悍将”,系统标准库中很多类都做了重构以支持异步序列。我们还可以用 AsyncStream 辅助结构非常方便的创建自己的异步序列。

在这里插入图片描述

这里我们就来一起聊聊 AsyncStream 结构,以及它新增的 makeStream 构建器方法。

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

  • 概览
  • 1. AsyncStream 旧构造器的弊端
  • 2. 拯救者:新方法 makeStream!
  • 总结

而在下篇中,我们将再接再厉继续讨论异步序列在使用时可能产生的内存泄漏、无限循环等等那些的潜伏陷阱。

相信学完本系列课程后,大家会对 Swift 新异步并发模型中异步序列的正确使用有更为深刻的领悟。

那还等什么呢?Let‘s find out!!!😉


1. AsyncStream 旧构造器的弊端

在 Swift 中创建自定义异步序列有很多种“姿势”,其中一个常见的方法是使用 AsyncStream 结构,可以认为它是一个异步序列的辅助构造器:

在这里插入图片描述

我们知道异步序列中的核心和精髓就是它的 Continuation 对象,做一个“二次元卡哇伊”的比喻:如果异步序列是一只大螃蟹,则 Continuation 就是它肥得流油的“蟹黄”:

在这里插入图片描述

值得注意的是,不像 Swift 中其它连续体(Continuation)对象,AsyncStream.Continuation 支持可逃逸(escaping)特性。这就让它的使用灵活性更上了一个层次。

我们使用 AsyncStream 创建异步序列主要有两种场景,一种是直接在其创建时就“包办”固定好所有元素的产出,但这样做缺乏变数、比较“死板”:

let stream = AsyncStream(unfolding: {
    return Int.random(in: 0..<Int.max)
})

另一种场景多半被用在 Apple 开发中的代理(Delegate)模式中,这种方式更加灵动自如:

protocol NumberSpawnerDelegate {
    func spawn(_ numbers: [Int])
}

struct Spawner {
    let timer = Timer.publish(every: 1.0, on: .main, in: .common).autoconnect()
    var delegator: NumberSpawnerDelegate?
    var cancel: Cancellable?
    
    mutating func setup() {
        cancel = timer.sink { [self] _ in
            var numbers = [Int]()
            for _ in 0..<Int.random(in: 1...3) {
                numbers.append(Int.random(in: 0...10000))
            }
            self.delegator?.spawn(numbers)
        }
    }
}

class AsyncNumberStream: NumberSpawnerDelegate {
    var continuation: AsyncStream<Int>.Continuation?
    
    lazy var stream: AsyncStream<Int> = {
        AsyncStream { continuation in
            self.continuation = continuation
        }
    }()
    
    func spawn(_ numbers: [Int]) {
        for i in numbers {
            continuation?.yield(i)
        }
    }
}

如上代码所示,我们的 AsyncNumberStream 异步序列遵从于 NumberSpawnerDelegate 协议,而 Spawner 作为驱动者自然就成为了 AsyncNumberStream 的事件源,它通过调用协议中的 spawn(😃 方法连接了发布者和接受者,使得天堑变通途。

我们可以这样使用 AsyncNumberStream 异步序列:

Task {
    let stream = AsyncNumberStream()
    var spawner = Spawner()
    spawner.delegator = stream
    spawner.setup()
    
    for await i in stream.stream {
        print("\(i)")
    }
}

运行结果如下所示:

在这里插入图片描述

不过这种以 AsyncStream 构造器“抓取”其 Continuation 对象的方式略显别扭(合肥话叫“肘手”)。而且 continuation 属性类型需要设置为可选值(AsyncStream<Int>.Continuation?),这多少让人觉得有些“不畅快”。

2. 拯救者:新方法 makeStream!

从 iOS 17.0 开始 Apple 为 AsyncStream 添加了一个新的 makeStream 方法专门用来解决上述窘境:

在这里插入图片描述

值得注意的是,虽然 makeStream 在 iOS 17 才被加入,但它向后兼容旧的系统(iOS 13 - iOS 17),所以在之前的 iOS 中也可以任性的使用它。

该方法返回一个由异步序列和其对应连续体组成的元组:

@available(macOS 10.15, iOS 13.0, watchOS 6.0, tvOS 13.0, *)
    @_backDeploy(before: macOS 14.0, iOS 17.0, watchOS 10.0, tvOS 17.0)
    public static func makeStream(of elementType: Element.Type = Element.self, bufferingPolicy limit: AsyncStream<Element>.Continuation.BufferingPolicy = .unbounded) -> (stream: AsyncStream<Element>, continuation: AsyncStream<Element>.Continuation)

这意味着之前“肘手”的调用可以改成这样:

class AsyncNumberStream: NumberSpawnerDelegate {
    let stream: AsyncStream<Int>
    private let continuation: AsyncStream<Int>.Continuation
    
    init() {
        let (stream, continuation) = AsyncStream.makeStream(of: Int.self)
        self.stream = stream
        self.continuation = continuation
    }
    
    func spawn(_ numbers: [Int]) {
        for i in numbers {
            continuation.yield(i)
        }
    }
}

从上面代码可以看到,AsyncStream.makeStream 方法带来了如下一些改变:

  • Continuation 不再“嵌入”在 AsyncStream 构造器的回调闭包之中,它们现在处在同一个层级;
  • continuation 属性不再要求是可选类型了;
  • 整体实现更加简单、一目了然;

现在,我们对 AsyncStream.Continuation 的获取不再聱牙诘屈,同时也完美的消除了 continuation 属性可选类型的限制,正谓是一举两得、一石二鸟也!

当然,可能有的小伙伴们觉得 AsyncStream.makeStream 方法如下形式的调用更加 nice 一些:

init() {
	let result = AsyncStream.makeStream(of: UUID.self)
	locations = result.stream
	continuation = result.continuation
}

值得一提的是,尽管我们将 AsyncNumberStream 内部的逻辑“粉饰一新”,但外部接口并没有丝毫改变。所以,之前的调用无需做任何修改。

编译运行代码可以发现,一切都未曾改变,正所谓平平淡淡才是真!棒棒哒!

虽然新的 makeStream 方法让我们原有的实现“清风徐来,水波不兴”,但异步序列本身的使用仍然暗影重重、波诡云谲。康庄大道上还有很多陷阱等着算计我们,我们将在下篇博文中将它们一网打尽!

总结

在本篇博文中,我们讨论了 Swift 5.5 新并发模型中用 AsyncStream 结构创建异步序列的新方法,并比较了它和之前旧的实现有哪些进步。

在下篇博文中,我们将继续异步序列的填坑之旅,期待吧!

感谢观赏,再会!😎

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

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

相关文章

海盾特种阀门诚邀您到场参观2024第13届生物发酵展

参展企业介绍 浙江海盾特种阀门有限公司是位于“中国泵阀之都”浙江温州&#xff0c;是一家集研发、生产、销售、服务于一体的专业流体控制阀生产企业&#xff0c;公司创立于1999年&#xff0c;公司一直秉承“创新是企业的发展之本&#xff0c;质量是企业的生存之本”的经营理…

数字图像处理项目——基于BCNN和迁移学习的鸟类图像细粒度分类(论文/代码)

完整的论文代码见文章末尾 以下为核心内容 摘要 本文采用了ResNet50、VGG19、InceptionV3和Xception等四种不同的深度神经网络模型&#xff0c;并应用于鸟类图像的细粒度分类问题中&#xff0c;以探究其在该任务上的性能表现。 其中&#xff0c;本文使用了BCNN&#xff08;B…

分布式锁的原子性问题

4.6 分布式锁的原子性问题 更为极端的误删逻辑说明&#xff1a; 线程1现在持有锁之后&#xff0c;在执行业务逻辑过程中&#xff0c;他正准备删除锁&#xff0c;而且已经走到了条件判断的过程中&#xff0c;比如他已经拿到了当前这把锁确实是属于他自己的&#xff0c;正准备删…

SysTick滴答定时器 - 延时函数

SysTick定时器 Systick定时器&#xff0c;是一个简单的定时器&#xff0c;对于CM3,CM4内核芯片&#xff0c;都有Systick定时器。Systick定时器常用来做延时&#xff0c;或者实时系统的心跳时钟。这样可以节省MCU资源&#xff0c;不用浪费一个定时器。比如UCOS中&#xff0c;分…

【PPT技巧】如何取消PPT的密码保护?

PPT文件有两种密码&#xff0c;一种是打开密码、一种是修改权限。今天分享这两种密码如何取消。 首先需要告知大家的是&#xff0c;密码的取消需要输入正确的密码。 打开密码的取消&#xff0c;我们需要先输入密码&#xff0c;打开文件&#xff0c;然后点击文件 – 信息 – 保…

贪心算法|135.分发糖果

力扣题目链接 class Solution { public:int candy(vector<int>& ratings) {vector<int> candyVec(ratings.size(), 1);// 从前向后for (int i 1; i < ratings.size(); i) {if (ratings[i] > ratings[i - 1]) candyVec[i] candyVec[i - 1] 1;}// 从后…

c++的学习之路:17、stack、queue与priority_queue

摘要 本文主要是介绍一下stack、queue、priority_queue的使用以及模拟实现&#xff0c;文章末附上代码以及思维导图。 目录 摘要 一、stack的介绍和使用 1、stack的介绍 2、stack的使用 3、stack的模拟实现 二、queue的介绍和使用 1、queue的介绍 2、queue的使用 3、…

Vue-B站学习笔记

1. 路由配置 B站视频之Vue route文件下的index.js app.vue

Nginx服务 重写功能与反向代理

六、重写功能 rewrite Nginx服务器利用 ngx_http_rewrite_module 模块解析和处理rewrite请求&#xff0c;此功能依靠 PCRE(perl compatible regular expression)&#xff0c;因此编译之前要安装PCRE库&#xff0c;rewrite是nginx服务器的重要功能之一&#xff0c;用于实现URL的…

土耳其航空2023年共运送旅客8340万人次,境内境外航线运力稳步增长

2023年,尽管面对持续紧张的国际局势和摇摆不定的宏观经济,土耳其航空仍实现了里程碑式的业绩表现,共计运输旅客8340万人次。土耳其境内航线运力比2022年增长了23.5%,运送旅客突破3000万人次;国际航线运力增长16%,运送旅客达5300万人次,并实现了14%的同比增长。其中,来自欧洲国家…

【Leetcode】2009. 使数组连续的最少操作数

文章目录 题目思路代码复杂度分析时间复杂度空间复杂度 结果总结 题目 题目链接&#x1f517; 给你一个整数数组 n u m s nums nums 。每一次操作中&#xff0c;你可以将 n u m s nums nums 中 任意 一个元素替换成 任意 整数。 如果 n u m s nums nums 满足以下条件&…

开创加密资产新纪元:深度解析ERC-314协议

随着加密资产市场的不断发展和区块链技术的日益成熟&#xff0c;新的协议和标准不断涌现&#xff0c;其中包括了ERC-314协议。本文将深入分析ERC-314协议的特点、功能以及对加密资产市场可能产生的影响。 1. ERC-314协议简介 ERC-314协议是一项建立在以太坊区块链上的新提案&a…

将精益IT和ITSM相结合,为客户和员工创造价值

在被要求使用精益 IT 之前&#xff0c;我已经从事IT服务管理 (ITSM)、运营、高级 IT 领导和顾问工作 30 多年了。那么什么是精益IT呢&#xff1f; 我的首席信息官派我去巴黎参加由 Steve Bell 和 Mike Orzen&#xff08;精益 IT 的作者&#xff09;主持的第一届精益 IT 会议。我…

使用ADO.NET访问数据库

目录 访问数据库的步骤 &#xff11;、建立数据库 &#xff12;、设置链接参数 &#xff08;1&#xff09;web网页和数据库连接的方法一 &#xff08;2&#xff09;web网页和数据库连接的方法二 &#xff13;、建立链接对象 &#xff14;、显示数据库 &#xff15;、数…

el-table 表格分页序号问题

el-table-column 本身提供typeindex 但是每次切换分页 序号都是1-9 想要获取当前页数的条数序号 获取不到 <el-table-column label"序号" width"60" align"center"><template #default"scope"><div>{{ (queryPara…

数据库之DQL操作(数据查询语言)

DQL英文全称是Data Query Language(数据查询语言)&#xff0c;数据查询语言&#xff0c;用来查询数据库中表的记录。查询关键字: SELECT。 本节介绍以下表为例&#xff1a; create table emp(id int comment 编号&#xff0c;workno varchar(10) comment 工号&#xff0c;nam…

解决npm install安装node-sass包容易失败的问题

具体问题如下&#xff1a; npm ERR! code ERESOLVE npm ERR! ERESOLVE unable to resolve dependency tree npm ERR! npm ERR! While resolving: XXX3.4.0 npm ERR! Found: webpack5.31.2 npm ERR! node_modules/webpack npm ERR! peer webpack”^4.0.0 || ^5.0.0″ from html-…

学生台灯护眼灯哪个牌子好?护眼台灯推荐儿童必备

当前&#xff0c;青少年及儿童的近视率居高不下&#xff0c;实际数据甚至超出了半数的比例。这意味着超过一半的孩子们可能面临着某种程度的视力问题&#xff0c;这一现象无疑让每位家长的心都紧绷起来&#xff0c;对孩子的视力健康格外关注。尤其是孩子学习时使用的那一盏台灯…

心灵鸡汤之励志正能量文案,积极向上热爱生活短句

1、在一切变好之前&#xff0c;我们总要经历一些不开心的日子&#xff0c;这段日子也许很长&#xff0c;也许只是一觉醒来。有时候&#xff0c;选择快乐&#xff0c;更需要勇气。 2、靠自己&#xff0c;才能无惧艰难&#xff0c;靠他人&#xff0c;永远害怕风霜&#xff0c;别…

【第二十七篇】几款配合Burpsuite使用的Google插件(Wappalyzer、FindSomething、FOFAproView等)

文章目录 WappalyzerFindSomethingFOFA Pro ViewsuperSearchPluswayback machinesource detecotorX-Forwarded-For Header以下插件均在Google应用商店中下载 Wappalyzer Wappalyzer是一个用于识别网站所使用的技术和工具的浏览器扩展程序。它能够检测出网站所使用的内容管理系…