首页 > 解决方案 > 引用命令替换的正确方法

问题描述

我有一个简单的 bash 脚本,它只输出作为位置参数提供给脚本的文件名:

#!/usr/bin/env bash

for file; do
    echo "$file"
done

假设我有带空格的文件(例如“f 1”和“f 2”)。我可以使用通配符调用脚本并获得预期的输出:

$ ./script f*
> f 1
> f 2

但是如果我使用命令替换它不起作用:

$ ./script $(echo f*)
> f
> 1
> f
> 2

当我的命令替换输出多个带空格的文件名时,如何正确引用?

编辑:我最终想要的是以随机顺序将文件名传递给脚本(这比仅仅回显它们的名称稍微复杂一些),例如:

./script $(ls f* | shuf)

标签: bashcommand-substitution

解决方案


使用 GNUshuf和 Bash 4.3+:

readarray -d '' files < <(shuf --zero-terminated --echo f*)
./script "${files[@]}"

其中--zero-terminated可以处理任何文件名,并且readarray还使用空字节作为分隔符。

对于readarray不支持该-d选项的旧 Bash:

while IFS= read -r -d '' f; do
    files+=("$f")
done < <(shuf --zero-terminated --echo f*)
./script "${files[@]}"

在有很多文件的极端情况下,这可能会遇到命令行长度限制;在这种情况下,

shuf --zero-terminated --echo f*

可以替换为

printf '%s\0' f* | shuf --zero-terminated

向Socowi指出的帽子提示--echo


推荐阅读