在循环中比较Bash中的小数

时间:2015-07-12 20:26:08

标签: bash floating-point shellcheck

在下面的代码中,ShellCheckwhile子句中引发错误。

count=10.0
while [ $count -le 20.0 ]
do
    echo "Hello"
    count=$(bc<<< "scale=4; (count+0.1)")
done

ShellCheck说:

  

不支持小数,使用整数或bc

我不太确定如何在while循环中使用bc。

while [ $(bc <<< "scale=4; (count -le 20.0)" ]

如何比较while子句中的十进制数?有什么建议吗?

3 个答案:

答案 0 :(得分:2)

Bash不支持浮点运算。 你可以使用bc:

count="10.0"
limit="12.0"
increment="0.1"

while [ "$(bc <<< "$count < $limit")" == "1"  ]; do
    echo "Hello"
    count=$(bc <<< "$count+$increment")
done

或awk:

while awk 'BEGIN { if ('$count'>='$limit') {exit 1}}'; do
    echo "Hello"
    count=$(bc <<< "$count+$increment")
done

我只是想知道:为什么不(直接)从10.0计算到12.0?

for i in $(seq 10.0 0.1 12.0); do
    echo "Hello"
done

答案 1 :(得分:1)

Bash不支持浮点运算。您也可以使用bc进行比较:

count=10.0

while : ;
do
    out=$(bc -l<<< "$count<20.0")
    [[ $out == 0 ]] && { echo "Reached limit" ; exit 0; }

    echo "Hello"
    count=$(bc<<< "scale=4; ($count+0.1)")
done

请注意,我在您更新$的循环中添加了遗失的countcount

答案 2 :(得分:0)

虽然bash不处理浮点数,但seq实用程序可以。 [注1]

基本语法为seq FIRST INCREMENT LAST,因此在您的情况下,您可以使用

for count in "$(seq 10.0 0.1 20.0)"; do
  # something with $count
done

如果你提供两个参数,它们被假定为FIRST和LAST,INCREMENT为1.如果你只提供一个参数,则假定它是LAST,FIRST和INCREMENT都是1.如你的例子所示,序列是包含,因此只要INCREMENT平均分配FIRST-LAST,就会生成FIRST和LAST。

您还可以包含明确的格式:

$ seq -f "%06.3f" 1 .5 2
01.000
01.500
02.000

这种技术的一个缺点是它预先计算了整个值集合。如果循环将执行数十万次,那可能会占用大量内存,在这种情况下,您可以使用管道或进程替换:

while read count; do
  # something with count
done < <(seq 10.0 0.000001 20.0)

注释

  1. seq不是Posix,但它几乎总是存在;它是GNU coreutils的一部分,类似的实用程序,在Mac OS X中可用)自3.0以来一直在NetBSD和9.0以来的FreeBSD。