比较 Bash while 循环中的小数

Posted

技术标签:

【中文标题】比较 Bash while 循环中的小数【英文标题】:Compare Decimals in Bash while Loop 【发布时间】:2015-10-01 01:25:04 【问题描述】:

在下面的代码中,ShellCheck 在while 子句中抛出错误。

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 子句中的十进制数?有什么建议吗?

【问题讨论】:

【参考方案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

请注意,我在更新count 的循环内将缺少的$ 添加到count

【讨论】:

【参考方案2】:

虽然 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)

注意事项

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

【讨论】:

【参考方案3】:

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

【讨论】:

以上是关于比较 Bash while 循环中的小数的主要内容,如果未能解决你的问题,请参考以下文章

您如何专门获得一个浮点数为 2 位小数的输入作为 C++ 中 while 循环的条件?

把 分数化为循环小数 和 把循环小数化为分数 的方法

循环小数——模拟除法

小数怎么转化分数?

20170803 - 今日技能封装 - A

数论证明(任何一个素数倒数的循环小数位数一定小于素数本身)