bash - 当 “nounset” 选项生效时获取空数组或未设置数组的长度

标签 bash

由于 Bash 在以 set -o nounset 模式(又名 set -u)运行时,可能会将空数组视为未设置,无论它们是否具有实际上被分配了一个空值,在尝试扩展数组时必须小心——其中一个解决方法是检查数组长度是否为零。更不用说获取数组中元素的数量本身就是一项常见操作。

在 openSUSE 42.1 中使用 Bash 4.2.47(1)-release 进行开发时,我习惯于在数组为空或未设置时使用 ${#ARRAY_NAME[@]} 获取数组大小成功.但是,在使用 FreeBSD 10.3 中的 Bash 4.3.46(1)-release 检查我的脚本时,结果发现此操作可能会失败并显示通用的“未绑定(bind)变量”错误消息。为扩展提供默认值似乎不适用于数组长度。 提供替代命令链似乎可行,但在通过子 shell 扩展调用的函数中不起作用——函数在第一次失败后退出。还有什么可以帮助您?

考虑以下示例:

function Size ()
{
    declare VAR="$1"
    declare REF="\${#${VAR}[@]}"
    eval "echo \"${REF}\" || echo 0" 2>/dev/null || echo 0
}

set -u
declare -a MYARRAY

echo "size: ${#MYARRAY[@]}"
echo "size: ${#MYARRAY[@]-0}"
echo "Size: $(Size 'MYARRAY')"
echo -n "Size: "; Size 'MYARRAY'

在 openSUSE 环境中,所有 echo 行都按预期输出 0。在 FreeBSD 中,只有当数组被显式分配一个空值时才会出现相同的结果:MYARRAY=();否则,前两行的内联查询均失败,第三行仅输出 Size:(意味着扩展结果为空),只有最后一行完全成功,这要归功于外部 || echo 0 — 然而,将结果传递到屏幕并不是尝试获取数组长度时的通常意图。

以下是我的观察总结:

                                    Bash 4.2  Bash 4.3
                                    openSUSE  FreeBSD

counting elements of unset array       OK      FAILED
counting elements of empty array       OK        OK

content expansion of unset array     FAILED    FAILED
content expansion of unset array(*)    OK        OK
content expansion of empty array     FAILED    FAILED
content expansion of empty array(*)    OK        OK
    (* with fallback value supplied)

对我来说,这看起来很不一致。是否有真正面向 future 的跨平台解决方案?

最佳答案

bash 的 Linux 和 BSD 风格之间存在已知(已记录)差异。我建议按照 POSIX 标准编写代码。您可以从这里开始了解更多信息 -> www2.opengroup.org。

考虑到这一点,您可以使用 --posix 命令行选项启动 bash,或者您可以执行命令 set -o posixbash 正在运行。两者都会导致 bash 符合 POSIX 标准。

以上建议会增加跨平台一致性的概率。

关于bash - 当 “nounset” 选项生效时获取空数组或未设置数组的长度,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38312825/

相关文章:

linux - 如果 bash 脚本的条件有什么问题

linux - 如何从 linux 中的分隔字符串中找到最小字符串?

用于同时 ping 多台主机的 Linux Bash 脚本

自定义 bash 函数的 Git 自动完成

linux - Bash - if 语句不会自动运行

bash - 并排连接两个文件,附加字段之间的差异,并以表格格式打印

linux - 如何在 Bash 中存储/保存命令的输出?

linux - 每次执行 bash 脚本时执行不同的任务

python - 如何在 json 文档中的单行 shell 脚本中制作一个大的 shell 脚本?

git - 用于检查当前 git 分支是否为 "x"的 bash 脚本