为什么 Bash 算术中的除法将大多数百分比计算为 0?

为什么 Bash 算术中的除法将大多数百分比计算为 0?

尝试对脚本进行 bash 运算,但$e直到最后才更新。输出不言而喻。

max=5
for e in $(seq 1 1 $max); do 
    percent=$(( $e/$max*100 ))
    echo "echo $e / $max : = $percent"
done

Tl;DR:以百分比显示 1..5。

输出 :

echo 1 / 5 : = 0
echo 2 / 5 : = 0
echo 3 / 5 : = 0
echo 4 / 5 : = 0
echo 5 / 5 : = 100

为什么是这样?

答案1

bash无法处理非整数算术。只要所有表达式都是整数,它就会给出正确的结果。因此,您需要避免在计算中的某个地方得到非整数值。

在您的情况下,当您评估等时1 / 52 / 5它会在 bash 中创建与某些非整数值相对应的整数零值,并且结果相应地为零。优先级除法和乘法的运算符号相同,并且相同的先行运算符总是按照它们在表达式中的位置从左到右执行。

一个解决方法是先做乘法,然后再做除法,这样 bash 就不必处理非整数值了。更正后的表达式将是,

$ max=5; for e in $(seq 1 1 $max); do percent=$(( $e*100/$max )); echo "echo $e / $max : = $percent"; done
echo 1 / 5 : = 20
echo 2 / 5 : = 40
echo 3 / 5 : = 60
echo 4 / 5 : = 80
echo 5 / 5 : = 100

答案2

Bash 在这种算术方面做得不是很好……这是你的问题:

$ echo $((1/5))
0
$ echo $((2/5))
0
$ echo $((4/5))
0
$ echo $((4/5))
0

如果需要处理非整数值,可以使用bc

$ max=5; for e in $(seq 1 1 "$max"); do percent=$(bc <<< "scale=1 ; $e/$max*100") ; echo "echo $e / $max : = ${percent%.*}"; done
echo 1 / 5 : = 20
echo 2 / 5 : = 40
echo 3 / 5 : = 60
echo 4 / 5 : = 80
echo 5 / 5 : = 100

(感谢@Arronical 指出如何将输出格式化为整数)

答案3

与 bash 不同,awk 提供完整的浮点运算。例如:

$ awk -v max=5 'BEGIN{for (e=1;e<=max;e++) print "echo " e " / " max " : = " 100*e/max}' 
echo 1 / 5 : = 20
echo 2 / 5 : = 40
echo 3 / 5 : = 60
echo 4 / 5 : = 80
echo 5 / 5 : = 100

答案4

尝试

percent=$(( $e*100/$max ))

:)

请参阅算术评估部分:

man bash

它仅支持整数。

相关内容