首页 > 解决方案 > 避免在 bash 中使用数组进行通配符扩展

问题描述

我写了以下代码:

join(){
        IFS="$1"
        shift
        echo "$*"
}

FILES=(/tmp/*)
SEPARATED_FILES=$(join , ${FILES[*]})
echo $VAR

它可以很好地打印逗号分隔的文件列表/tmp。但我想重构它并消除FILES作为数组的 tmp 全局变量。我尝试了以下方法:

SEPARATED_FILES=$(join , ${(/tmp/*)[*]})
echo $VAR

但它会打印以下错误:

line 8: ${(/tmp/*)[*]}: bad substitution    

标签: linuxbash

解决方案


是的!您可以通过将 glob 作为参数直接传递给函数来避免它。请注意,glob 结果在传递给函数之前由 shell 扩展。因此,将第一个参数作为IFS您要设置的参数,将第二个参数作为您要使用的 glob 表达式。

join , /tmp/*

在调用函数之前,将 glob 扩展为文件名。

join , /tmp/file1 /tmp/file2 /tmp/file3

上面的一个值得注意的补充是在调用函数之前使用nullglob选项。因为当 glob 不产生任何结果时,可以安全地忽略未展开的字符串。

shopt -s nullglob
join , /tmp/*

并在命令替换语法中为

fileList=$(shopt -s nullglob; join , /tmp/*)

从你的努力中得到了一些收获。

  1. 除非您有理由不这样做,否则始终将 shell 引用应用于变量/数组。这样做可以保留内部内容的字面值并防止发生分词
  2. 始终为用户定义的变量/函数和数组名称使用小写名称

推荐阅读