使用 coproc

问题描述

我对 linux shell 脚本很陌生,有一个问题:

1.) 为什么这个命令

test1="leafpad" && coproc exec "$test1"

在 bash 中工作(命令行,GNU bash 4.4.12 在 debian derivate linux 上),但是命令

test1="coproc" && exec "$test1" leafpad

不是吗?错误信息: bash: exec: coproc: 未找到。

coproc leafpad 确实按预期工作。

此命令必须如何正确引用才能使其工作?我已经试过了

test1=`coproc` && exec "$test1" leafpad

test1='coproc' && exec "$test1" leafpad

test1="'coproc'" && exec "$test1" leafpad

test1=`coproc` && exec '$test1' leafpad

test1=`coproc` && exec `$test1` leafpad

test1="coproc" && exec $test1 leafpad

test1=`coproc` && exec $test1 leafpad

还有一些变体,但它们都不起作用。

2.) 这只是在命令行上的测试。但我更需要的是在脚本中执行此操作:所以我确定需要做一些额外的引用或伪装特殊字符。

背景: 我必须执行一个命令,其中包含许多参数,其中一些被变量替换。想想像 yad 这样的东西,在几行中包含所有可能的参数,但让我们创建一个更简单的例子:

my_codeset="437"
my_line="20"
my_filename="somthing.txt"
if [ $value == 0 ]; then my_tabwidth='--tab-width=10'; else my_tabwidth=""; fi   # set tabs or not

leafpad --codeset="$my_codeset" "$my_tabwidth" --jump="$my_line" "$my_filename"

其中给定的变量是变化的主题,作为之前用户交互的函数

在这个完整的命令(原文大约6行代码),需要分两种执行: 一次由 coproc 领导,另一次不是,作为条件分支的函数

所以我想要的是:

if [ $something == 1 ]; then copr_this="coproc"; else copr_this=""; fi

exec '$copr_this' <mycommand,with all its quoted arguments>

代替

if [ something == 0]; then
          lengthy command here
else
          coproc lengthy command here,exactly repeated.
fi

我已经尝试以相反的方式管理它,即将完整冗长的命令放在一个变量中,并在条件分支中执行它:

my_command=`lengthy command with some arguments $arg1 $arg2 ...`

if...
      exec "$my_command"
else
      coproc exec "$my_command"
fi

也停止了错误消息“未找到”。不同的引用方式并没有解决它,只是产生了不同的错误信息。我没能找到这个任务的正确引用。如何正确阅读此报价?

我当然可以重复代码中的 6 行命令,但我很确定这样做会更方便。

如开头所述:间接命令执行适用于命令行(也适用于脚本),只要不涉及 coproc。我无法让它与 coproc 一起工作。

感谢任何帮助和提示


@Socowi 第一个回答后更新:

感谢您提供全面而快速的答复,Socowi。您显然对 coproc 不是命令是正确的。所以我现在明白为什么我的尝试必须失败。 exec 命令仅在我的实验期间添加。我开始时没有这个,但在没有成功之后我认为它会有所帮助。这只是一种绝望的行为。 my_command=`lengthy command with some arguments $arg1 $arg2 ...` 行中的反引号是一个错字,应该有正常的引号,正如您所指出的,因为我当然打算在 if 中执行命令。我可能会按照您指示的方式前往,在脚本中使用 function {...}。但是在此期间对这个问题进行了实验,我得出了一个惊人的解决方案:对我来说令人惊讶,因为 coproc 不是命令和 leafpad间的区别是二进制命令。所以应该清楚 test1='coproc' && test2='leafpad' && "$test1 $test2" 将失败并显示错误消息 bash: coproc leafpad: command not found.,这是真的。但是现在:为什么 test1='coproc' && test2='leafpad' && /bin/bash -c "$test1 $test2" 会完成这项工作,启动 Leafpad,允许在 bash 并行中输入更多命令,就像我只输入了 leafpad & 一样?但是这一次同时执行,内置(或关键字?)和命令,从一个变量,当试图直接在第一个 bash 实例中输入它时被拒绝。对于 bash 的第一个实例,第二个实例也应该是真的,还是我有错误的观点?为什么它以这种方式工作? -c 选项除了执行命令之外还有其他作用吗?

解决方法

引用不是这里的问题。还有两个问题:

execcoproc 以及内置函数与二进制文件的顺序

test1="leafpad" && coproc exec "$test1"coproc exec leafpad 相同。
test1="coproc" && exec "$test1" leafpadexec coproc leafpad 相同。

顺序有所不同:coproc execexec coproc。后者不起作用,因为 exec 用指定的程序替换了当前的 shell。但是,coproc 是一个内置命令。您的系统上没有 coproc 二进制文件。您只能从 bash 内部运行它。因此 exec 失败。

命令替换与字符串

在你的脚本中......

my_command=`lengthy command`
if ...; then
      exec "$my_command"
else
      coproc exec "$my_command"
fi

... 您没有将 lengthy command 存储在变量中,但是您运行了该命令并在 {{1} 之前存储了其输出(v=`cmd` v=$(cmd) 相同) }.然后在 if 中,您尝试将命令的输出作为另一个命令执行。

要将命令存储为字符串并稍后执行,您可以使用 if(注意故意丢失的引号)。然而,bash 提供了更好的方法来存储命令。使用数组或函数代替字符串。这里我们使用一个函数:

my_command="lengthy command"; $my_command

my_command() { exec lengthy command } if ...; then coproc my_command else my_command fi

话虽如此,我真的很想知道 coproc execcoproc 的组合。对我来说,exec 似乎忽略了 coproc exec cmd 部分并且与 exec 相同。如果 coproc cmd 在这里正常运行,当前的 shell 将被替换,您将丢失 exec 数组,因此不需要 COPROC。无论哪种方式,同时使用两者似乎很奇怪。你真的确定你需要那个 coproc 吗?如果是这样,我很乐意听到原因。