为什么这段代码可以在 bash v4.4 中运行,但不能在 bash v3.2 中运行?

为什么这段代码可以在 bash v4.4 中运行,但不能在 bash v3.2 中运行?

我有以下 bash 脚本:

#!/bin/bash

encl0=( 0,0 0,1 0,2 0,3 0,4 0,5 0,7 0,8 0,9 0,10 0,11 0,12 0,13 0,14 0,15 )
MISSING_DISKS=()
OLDIFS=$IFS
IFS=$'\n'
MISSING_DISKS+=($({ printf '0 %s\n' {0..15}; printf '0 %s\n' "${encl0[@]#0,}"; } | sort | uniq -u))
IFS=$OLDIFS
echo "$({ printf '0 %s\n' {0..15}; printf '0 %s\n' "${encl0[@]#0,}"; } | sort | uniq -u)"
echo "${MISSING_DISKS[@]}"
if ((${#MISSING_DISKS[@]}>1)); then
    echo "Greater than 1"
else
    echo "Success"
fi

当我使用 bash v4.4 运行它时,它按我的预期工作:

$ /usr/local/bin/bash test.sh
0 6
0 6
Success

但是,当我使用 bash v3.2 运行它时,它不会:

$ /bin/bash test.sh
0 6
0 0 0 0 1 2 3 4 5 7 8 9 10 11 12 13 14 15 0 1 0 10 0 11 0 12 0 13 0 14 0 15 0 2 0 3 0 4 0 5 0 6 0 7 0 8 0 9
Greater than 1

我不明白如何MISSING_DISKS将其设置为与设置它的命令的输出不同的东西。有谁知道是什么原因造成的?

答案1

使用空格、制表符或换行符总是接近失败。

核心问题出现在这里:

encl0=( 0,0 0,1 0,2 0,3 0,4 0,5 0,7 0,8 0,9 )
IFS=$'\n'
printf '<0 %s>\n' "${encl0[@]#0,}"

如果在 bash 3.2 上执行:

$ b32sh ./script
<0 0 1 2 3 4 5 7 8 9>

的扩展"${encl0[@]#0,}"作为一个字符串而不是值列表进行处理。

如果 IFS 有空格或者扩展未编辑数组的每个值,则不会出现此问题:

#!/bin/bash
encl0=( 0,0 0,1 0,2 0,3 0,4 0,5 0,7 0,8 0,9 )
IFS=$' \n'
printf '<0 %s>\n' "${encl0[@]#0,}"

执行:

$ b32sh ./script
<0 0>
<0 1>
<0 2>
<0 3>
<0 4>
<0 5>
<0 7>
<0 8>
<0 9>

或者:

#!/bin/bash
encl0=( 0,0 0,1 0,2 0,3 0,4 0,5 0,7 0,8 0,9 )
IFS=$'\n'
printf '<0 %s>\n' "${encl0[@]}"

执行:

b32sh ./so
<0 0,0>
<0 0,1>
<0 0,2>
<0 0,3>
<0 0,4>
<0 0,5>
<0 0,7>
<0 0,8>
<0 0,9>

该问题隐藏在您的脚本中,因为您IFS=$OLDIFS 在测试回显行之前恢复了 IFS。

避免此问题的一种方法是不在 printf 中使用空格:

#!/bin/bash
encl0=( 0,0 0,1 0,2 0,3 0,4 0,5 0,7 0,8 0,9 0,10 0,11 0,12 0,13 0,14 0,15 )
MISSING_DISKS=()
OLDIFS=$IFS
IFS=$' \n'
MISSING_DISKS+=($({ printf '0x%s\n' {0..15}; printf '0x%s\n' "${encl0[@]#0,}"; } | sort | uniq -u))
echo "test $({ printf '0x%s\n' {0..15}; printf '0x%s\n' "${encl0[@]#0,}"; } | sort | uniq -u)"
echo "var  ${MISSING_DISKS[@]}"
if ((${#MISSING_DISKS[@]}>1)); then
    echo "Greater than 1"
else
    echo "Success"
fi
IFS=$OLDIFS

另一种选择是通过使用备用数组将 IFS 更改为换行符后避免替换扩展:

#!/bin/bash
OLDIFS=$IFS
encl0=( 0,0 0,1 0,2 0,3 0,4 0,5 0,7 0,8 0,9 0,10 0,11 0,12 0,13 0,14 0,15 )
IFS=$' \n'; arr=("${encl0[@]#0,}")
MISSING_DISKS=()
IFS=$'\n'
MISSING_DISKS+=($({ printf '0 %s\n' {0..15}; printf '0 %s\n' "${arr[@]}"; } | sort | uniq -u))
echo "test $({ printf '0 %s\n' {0..15}; printf '0 %s\n' "${arr[@]}"; } | sort | uniq -u)"
echo "var  ${MISSING_DISKS[@]}"
if ((${#MISSING_DISKS[@]}>1)); then
    echo "Greater than 1"
else
    echo "Success"
fi
IFS=$OLDIFS

我建议你:

  • 遵循以下规则:CAPS 中的变量是环境变量。
  • 无需增加+=代码中此时为空的数组MISSING_DISKS+=
  • 在 printf 中使用非空格字符以避免后面需要从 IFS 中删除空格。这使得脚本更加健壮。

如果完成这些更改,脚本将变为:

#!/bin/bash
oldIFS=$IFS
encl0=( 0,0 0,1 0,2 0,3 0,4 0,5 0,7 0,8 0,9 0,10 0,11 0,12 0,13 0,14 0,15 )
unset missing_disks

IFS=' '
           arr=($(printf '0-%s\n' "${encl0[@]#0,}"))
           arr+=($(printf '0-%s\n' {0..15}))

missing_disks=($(printf '%s\n' "${arr[@]}" | sort | uniq -u))

           echo "test $(printf '0-%s\n' "${arr[@]}" | sort | uniq -u)"
           echo "var  ${missing_disks[@]}"

((${#missing_disks[@]}>1)) && echo "Greater than 1" || echo "Success"
IFS=$oldIFS

相关内容