为什么不使用 GlobalScope.launch?

2023-11-21

我读到了这个用法Globalscope非常沮丧,here.

我有一个简单的用例。对于我收到的每条 kafka 消息(假设是一个 Id 列表),我必须将其拆分并同时为每个 Id 调用休息服务,并等待其完成并继续执行其他同步任务。该应用程序中没有其他内容需要协程。在这种情况下,我可以使用Globalscope ?

注意:这不是 Android 应用程序。它是一个运行在服务器端的kafka流处理器。它是一个在 Kubernetes 中运行的临时、无状态、容器化 (Docker) 应用程序(如果您愿意,也可以符合 Buzzword 标准)


您应该使用结构化并发来适当地确定并发范围。如果您不这样做,您的协程可能会泄漏。就您而言,将它们的范围限定为单个消息的处理似乎是合适的。

这是一个例子:

/* I don't know Kafka, but let's pretend this function gets 
 * called when you receive a new message
 */
suspend fun onMessage(msg: Message) {
    val ids: List<Int> = msg.getIds()    

    val jobs = ids.map { id ->
        GlobalScope.launch { restService.post(id) }
    }

    jobs.joinAll()
}

如果其中一个调用restService.post(id)如果因异常而失败,该示例将立即重新抛出异常,并且所有尚未完成的作业都将泄漏。他们将继续执行(可能无限期地),如果他们失败,你不会知道。

为了解决这个问题,您需要确定协程的范围。这是没有泄漏的相同示例:

suspend fun onMessage(msg: Message) = coroutineScope {
    val ids: List<Int> = msg.getIds()    

    ids.forEach { id ->
        // launch is called on "this", which is the coroutineScope.
        launch { restService.post(id) }
    }
}

在这种情况下,如果其中一个调用restService.post(id)失败,那么协程范围内的所有其他未完成的协程将被取消。当您离开范围时,您可以确定您没有泄漏任何协程。

另外,因为coroutineScope将等到所有子协程完成,您可以删除jobs.joinAll() call.

边注: 编写启动某些协程的函数时的一个约定是让调用者使用接收者参数来决定协程范围。这样做与onMessage函数可能如下所示:

fun CoroutineScope.onMessage(msg: Message): List<Job> {
    val ids: List<Int> = msg.getIds()    

    return ids.map { id ->
        // launch is called on "this", which is the coroutineScope.
        launch { restService.post(id) }
    }
}
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

为什么不使用 GlobalScope.launch? 的相关文章

随机推荐

  • 是否可以像在数据网格中一样过滤 dgrid 中的数据?如果是这样,怎么办?

    我对 dojo 比较陌生 并且了解了 datagrid 如何提供动态过滤功能 该功能可以根据您在过滤器文本输入中键入的内容来减少可见行 我还没有找到任何关于如何使用 dgrid 执行此操作的示例 如果可以完成 请提供一个示例或向我指出提供教
  • C++删除txt文件中的最后一个字符

    我需要一些关于删除 txt 文件中最后一个字符的帮助 例如 如果我的txt文件包含1234567 我需要C 代码删除最后一个字符 使文件变成123456 谢谢大家 在可移植代码中执行此操作的唯一方法是读入数据 并写出除最后一个字符之外的所有
  • Windows 中的长路径/文件名会使 R 中的 write.table() 出错

    在 R 中 我使用的是write table 将文件写入嵌入长名称目录的位置 但是却报错如下 文件中的错误 文件 ifelse append a w 无法打开连接 另外 警告消息 在文件 文件 ifelse 追加 a w 中 无法打开文件
  • PHP 中检测浏览器连接关闭

    有谁知道是否可以检测浏览器在长时间执行过程中是否关闭了连接PHP脚本 当使用apache and mod php 例如 在Java the HttpOutputStream会抛出一个exception如果在浏览器关闭后尝试写入 或者会做出否
  • Python:从经验分布生成随机值

    在Java中 我通常依赖org apache commons math3 random EmpiricalDistribution类执行以下操作 从观察到的数据得出概率分布 根据该分布生成随机值 有没有提供相同功能的 Python 库 这好
  • SWIFT:为什么我无法在 UIWebView 中加载当前 URL? [关闭]

    Closed 这个问题是无法重现或由拼写错误引起 目前不接受答案 我需要将当前 URL 加载到 web 视图中 这就是我试图获取的方法 但它给了我这个错误 无法转换表达式类型 ST7 输入 字符串 这是代码 var currentURL N
  • 如何为 VS Code 创建简单的自定义语言着色

    我正在尝试为日志文件创建一个简单的着色 现在可以在代码中包含自定义语言 我使用的是 0 9 2 我创建了一个简单的 tmLanguage 文件来对字母 q 进行着色 只是为了启动 但没有成功 我的新语言 log 与文件扩展名正确关联 我也可
  • 如何查找 MySQL 中缺失的索引?

    5 我想识别其中丢失的索引 有人可以帮我识别吗 这将帮助我们提高导致应用程序的查询的性能 我能想到的最好的办法就是使用EXPLAIN检查带索引和不带索引的查询的执行计划 然后查看查询性能的差异 您还可以参考 MySQL执行计划是否依赖于可用
  • VS 更新到 17.7.3 后无法连接到 Mac

    我刚刚在 Windows PC 上安装了新的 Visual Studio 2022 Update 17 7 3 从那时起 我无法再与我的 Mac 配对来部署我的 NET MAUI 应用程序 我收到以下错误消息 无法安装工作负载 ios 详细
  • 我需要一个初学者指南来设置 Windows 进行 Python 开发

    我目前只使用 NET 并且想尝试一下 python 为此我需要搭建一个python开发环境 我对此的指导会很方便 我想我会进行网络开发 所以需要一个网络服务器 可能还需要一个数据库 我还需要流行的 ORM MVC 框架和测试库的指针 我对这
  • 使用 Jetpack Compose 时如何使两个按钮宽度相同?

    我运行代码 A 并得到结果 A 如何使两个按钮的宽度相同 顺便说一句 你知道不同的字符串有不同的宽度 你不能编写硬代码 例如Modifier width 100 dp Code A Row modifier Modifier fillMax
  • 如何上传/设置Azure Blob存储的default.html页面?

    首先 如何将文件上传到 blob 根目录 看来我必须将文件上传到容器 IMO 是一个文件夹 其次 如果我最终找到了将文件上传到根目录的方法 default html 页面 或其他名称 是否会成为该域的默认页面 Thanks 有一个名为 ro
  • 如何提取画布中图像的一部分并将其用作 div 的背景图像?

    这就是我的代码的样子 document addEventListener DOMContentLoaded function var canvas document querySelector canvas var ctx canvas g
  • 部分类、LINQ、接口和 VB.NET

    好的 我在 VB NET 中遇到了问题 那么所有 VB NET 的维护者们能帮帮我吗 这是我的问题 我正在使用 LINQ to Entities 它也可以与 LINQ to SQL 一起使用 我构建我的 edmx 文件 然后创建一个与实体之
  • traefik代理后面的docker中的gitlab失败(通常)

    我有几个网站在 docker 中运行 使用 LetsEncrypt 凭据并通过 traefik 进行路由 我想在 docker 中使用 LetsEncrypt 和 traefik 运行本地 gitlab ce 所以我将其添加到我的 trae
  • Typescript 接口默认值

    我在 TypeScript 中有以下界面 interface IX a string b any c AnotherType 我声明该类型的变量并初始化所有属性 let x IX a abc b null c null 然后我稍后在 ini
  • 读取csv文件中的特定行,python

    In an CSV使用python的文件我们可以逐行或逐行读取所有文件 我想读取特定行 第24行示例 而不读取所有文件和所有行 您可以使用行缓存 getline linecache getline 文件名 lineno module glo
  • 使用高级加密标准算法 (AES) 在 Typescript 中加密字符串并在 C# 中解密

    我很难在打字稿中实现加密并在 C 中实现解密 在在这里发布问题之前 我用 Google 搜索并找到了一些链接 但这些链接与 JavaScript 而不是打字稿相关 在 javascript 中加密并使用 AES 算法在 C 中解密 使用 a
  • 如何告诉 Eclipse 使用与通常不同的 JRE 版本来编译和构建项目?

    我不确定这个问题是否已得到完整回答 或者我的标题是否足够描述我的情况 但我被要求将项目从使用 Ant 构建转换为 Maven 这部分还不错 但我被告知这个应用程序是专门为 JRE 1 5 版而不是我一直在处理的其他应用程序使用的 JRE 6
  • 为什么不使用 GlobalScope.launch?

    我读到了这个用法Globalscope非常沮丧 here 我有一个简单的用例 对于我收到的每条 kafka 消息 假设是一个 Id 列表 我必须将其拆分并同时为每个 Id 调用休息服务 并等待其完成并继续执行其他同步任务 该应用程序中没有其