AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • 主页
  • 系统&网络
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • 主页
  • 系统&网络
    • 最新
    • 热门
    • 标签
  • Ubuntu
    • 最新
    • 热门
    • 标签
  • Unix
    • 最新
    • 标签
  • DBA
    • 最新
    • 标签
  • Computer
    • 最新
    • 标签
  • Coding
    • 最新
    • 标签
主页 / unix / 问题 / 531198
Accepted
kjo
kjo
Asked: 2019-07-21 08:16:15 +0800 CST2019-07-21 08:16:15 +0800 CST 2019-07-21 08:16:15 +0800 CST

是否可以有多个并发协同进程?

  • 772

下面的测试脚本1的目的是启动一个“外部”协同进程(正在运行),在 -loop 中从此协同进程中读取,并且对于读取的每一行,打印一行标识外部循环的当前迭代,启动一个“ inner" 协进程(也在运行,带有新参数),在嵌套的 while 循环中从这个内部协进程中读取,然后清理这个内部协进程。嵌套的 while 循环为它从内部协进程读取的每一行打印一些输出。seq 3whileseq

#!/bin/bash
# filename: coproctest.sh
PATH=/bin:/usr/bin

coproc OUTER { seq 3; }
SAVED_OUTER_PID="${OUTER_PID}"

exec {OUTER_READER}<&"${OUTER[0]}"
while IFS= read -r -u "${OUTER_READER}" OUTER_INDEX; do

    printf -- '%d\n' "${OUTER_INDEX}"

    START=$(( OUTER_INDEX * 1000000 ))
    FINISH=$(( START + OUTER_INDEX ))

    # (
      coproc INNER { seq "${START}" "${FINISH}"; }
      SAVED_INNER_PID="${INNER_PID}"
      exec {INNER_READER}<&"{INNER[0]}"

      while IFS= read -r -u "${INNER_READER}" INNER_INDEX; do
          printf -- '    %d\n' "${INNER_INDEX}"
      done

      exec {INNER_READER}<&-

      wait "${SAVED_INNER_PID}"
    # )

done
exec {OUTER_READER}<&-
wait "${SAVED_OUTER_PID}"

当我运行这个脚本时,这是我得到的输出:

% ./coproctest.sh
1
./coproctest.sh: line 30: warning: execute_coproc: coproc [12523:OUTER] still exists
./coproctest.sh: line 19: INNER_READER: ambiguous redirect
./coproctest.sh: line 21: read: : invalid file descriptor specification
./coproctest.sh: line 25: INNER_READER: ambiguous redirect
2
./coproctest.sh: line 19: INNER_READER: ambiguous redirect
./coproctest.sh: line 21: read: : invalid file descriptor specification
./coproctest.sh: line 25: INNER_READER: ambiguous redirect
3
./coproctest.sh: line 19: INNER_READER: ambiguous redirect
./coproctest.sh: line 21: read: : invalid file descriptor specification
./coproctest.sh: line 25: INNER_READER: ambiguous redirect

如果我取消注释两条注释行,我会得到几乎相同的输出。


Q1:是否可以同时运行多个协同进程?

Q2:如果是这样,应该如何修改上面的脚本以实现所需的输出?


1我最近才开始使用协同进程,还有很多我不明白的地方。因此,该脚本几乎肯定包含不正确、笨拙或不必要的代码。请随时评论和/或修复您的回复中的这些弱点。

bash shell-script
  • 2 2 个回答
  • 1223 Views

2 个回答

  • Voted
  1. Best Answer
    Kusalananda
    2019-07-21T08:17:59+08:002019-07-21T08:17:59+08:00

    bash从手册末尾的“BUGS”部分:

    一次可能只有一个活动的协同进程。

    • 5
  2. LL3
    2019-07-23T05:31:32+08:002019-07-23T05:31:32+08:00

    Q1:是否可以同时运行多个协同进程?

    正如@Kusalananda 所指出的,在 Bash v4 及更高版本(包括当前的 v5)上,正式没有。

    但是,我可以告诉你,尽管有警告,它可能会起作用,但当然不能保证和 YMMV。请参阅此处了解更多信息。

    Q2:如果是这样,应该如何修改上面的脚本以实现所需的输出?

    一旦您修复了以下问题,它可能(如上所述)工作得很好:

    exec {INNER_READER}<&"{INNER[0]}"  # <-- lacks the '$' sign for the 'INNER[0]' variable
    

    这导致:

    ./coproctest.sh: line 19: INNER_READER: ambiguous redirect
    

    消息,因此还有:

    ./coproctest.sh: line 21: read: : invalid file descriptor specification
    

    信息。

    它对我有用,一旦解决了这个问题并把警告放在一边。

    至于其他注意事项:

    • 无需复制 acoproc的文件描述符,除非您想将它们传递给子进程(子外壳或命令或脚本)
    • 您这样做可能是因为seq命令自然很快完成,因此自动变量在您使用它们之前就消失了。通过这样做,那些重复的文件描述符将被所有后续命令和后台进程继承,如果您的协同进程实际使用其输入管道并等待它关闭以退出,这可能是不可取的。因此,另一种解决此问题的方法是通过同步机制,例如让您的OUTER协同进程成为{ seq 3; exec >&-; read; },然后当您从主脚本消耗其输入时,例如echo >&${OUTER[1]}; wait "${OUTER_PID}"让协同进程read继续进行wait。请注意,不能保证wait将在$OUTER_PID变量消失:在这种情况下,您可以将警告消息静音(或完全忽略它),并可能使用|| true
    • 作为个人的旁注,我可以告诉你,如果你真的决定同时拥有多个协同进程,你可以重新实现一个粗略的等价物,即coproc在后台进程和命名 FIFO 上使用 Bash v3 语法mkfifo,在 Linux 上也是使用 Process Substitutions 而不是mkfifos 的技巧。使用 Bash v4 语法,它可以不那么复杂,但仍然是一个具有挑战性的练习。
    • 2

相关问题

  • 从文本文件传递变量的奇怪问题

  • 虽然行读取保持转义空间?

  • MySQL Select with function IN () with bash array

  • `tee` 和 `bash` 进程替换顺序

  • 运行一个非常慢的脚本直到它成功

Sidebar

Stats

  • 问题 205573
  • 回答 270741
  • 最佳答案 135370
  • 用户 68524
  • 热门
  • 回答
  • Marko Smith

    模块 i915 可能缺少固件 /lib/firmware/i915/*

    • 3 个回答
  • Marko Smith

    无法获取 jessie backports 存储库

    • 4 个回答
  • Marko Smith

    如何将 GPG 私钥和公钥导出到文件

    • 4 个回答
  • Marko Smith

    我们如何运行存储在变量中的命令?

    • 5 个回答
  • Marko Smith

    如何配置 systemd-resolved 和 systemd-networkd 以使用本地 DNS 服务器来解析本地域和远程 DNS 服务器来解析远程域?

    • 3 个回答
  • Marko Smith

    dist-upgrade 后 Kali Linux 中的 apt-get update 错误 [重复]

    • 2 个回答
  • Marko Smith

    如何从 systemctl 服务日志中查看最新的 x 行

    • 5 个回答
  • Marko Smith

    Nano - 跳转到文件末尾

    • 8 个回答
  • Marko Smith

    grub 错误:你需要先加载内核

    • 4 个回答
  • Marko Smith

    如何下载软件包而不是使用 apt-get 命令安装它?

    • 7 个回答
  • Martin Hope
    user12345 无法获取 jessie backports 存储库 2019-03-27 04:39:28 +0800 CST
  • Martin Hope
    Carl 为什么大多数 systemd 示例都包含 WantedBy=multi-user.target? 2019-03-15 11:49:25 +0800 CST
  • Martin Hope
    rocky 如何将 GPG 私钥和公钥导出到文件 2018-11-16 05:36:15 +0800 CST
  • Martin Hope
    Evan Carroll systemctl 状态显示:“状态:降级” 2018-06-03 18:48:17 +0800 CST
  • Martin Hope
    Tim 我们如何运行存储在变量中的命令? 2018-05-21 04:46:29 +0800 CST
  • Martin Hope
    Ankur S 为什么 /dev/null 是一个文件?为什么它的功能不作为一个简单的程序来实现? 2018-04-17 07:28:04 +0800 CST
  • Martin Hope
    user3191334 如何从 systemctl 服务日志中查看最新的 x 行 2018-02-07 00:14:16 +0800 CST
  • Martin Hope
    Marko Pacak Nano - 跳转到文件末尾 2018-02-01 01:53:03 +0800 CST
  • Martin Hope
    Kidburla 为什么真假这么大? 2018-01-26 12:14:47 +0800 CST
  • Martin Hope
    Christos Baziotis 在一个巨大的(70GB)、一行、文本文件中替换字符串 2017-12-30 06:58:33 +0800 CST

热门标签

linux bash debian shell-script text-processing ubuntu centos shell awk ssh

Explore

  • 主页
  • 问题
    • 最新
    • 热门
  • 标签
  • 帮助

Footer

AskOverflow.Dev

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve