我正在努力比较Bash脚本中的两个浮点数。我有两个变量,例如。
let num1=3.17648e-22
let num2=1.5
现在,我想对这两个数字做一个简单的比较:
st=`echo "$num1 < $num2" | bc`
if [ $st -eq 1]; then
echo -e "$num1 < $num2"
else
echo -e "$num1 >= $num2"
fi
不幸的是,我有一些问题与num1的正确处理可以是“电子格式”。
我正在努力比较Bash脚本中的两个浮点数。我有两个变量,例如。
let num1=3.17648e-22
let num2=1.5
现在,我想对这两个数字做一个简单的比较:
st=`echo "$num1 < $num2" | bc`
if [ $st -eq 1]; then
echo -e "$num1 < $num2"
else
echo -e "$num1 >= $num2"
fi
不幸的是,我有一些问题与num1的正确处理可以是“电子格式”。
当前回答
一个纯Bash解决方案,用于比较没有指数符号,前导或尾随零的浮点数:
if [ ${FOO%.*} -eq ${BAR%.*} ] && [ ${FOO#*.} \> ${BAR#*.} ] || [ ${FOO%.*} -gt ${BAR%.*} ]; then
echo "${FOO} > ${BAR}";
else
echo "${FOO} <= ${BAR}";
fi
逻辑运算符的顺序很重要。整数部分作为数字进行比较,小数部分故意作为字符串进行比较。使用这种方法将变量分成整数部分和小数部分。
它不会比较浮点数和整数(不带点)。
其他回答
支持所有可能的表示法的解决方案,包括大小写指数的科学表示法(例如,12.00e4):
if (( $(bc -l <<< "${value1/e/E} < ${value2/e/E}") ))
then
echo "$value1 is smaller than $value2"
fi
对于非整数数学,最好使用AWK。你可以使用这个Bash实用函数:
numCompare() {
awk -v n1="$1" -v n2="$2" 'BEGIN {printf "%s " (n1<n2?"<":">=") " %s\n", n1, n2}'
}
称之为:
numCompare 5.65 3.14e-22
5.65 >= 3.14e-22
numCompare 5.65e-23 3.14e-22
5.65e-23 < 3.14e-22
numCompare 3.145678 3.145679
3.145678 < 3.145679
我使用这里的答案,并把它们放在一个函数中。你可以这样使用它:
is_first_floating_number_bigger 1.5 1.2
result="${__FUNCTION_RETURN}"
一旦被调用,echo $result在这种情况下将为1,否则为0。
功能:
is_first_floating_number_bigger () {
number1="$1"
number2="$2"
[ ${number1%.*} -eq ${number2%.*} ] && [ ${number1#*.} \> ${number2#*.} ] || [ ${number1%.*} -gt ${number2%.*} ];
result=$?
if [ "$result" -eq 0 ]; then result=1; else result=0; fi
__FUNCTION_RETURN="${result}"
}
或者带有调试输出的版本:
is_first_floating_number_bigger () {
number1="$1"
number2="$2"
echo "... is_first_floating_number_bigger: comparing ${number1} with ${number2} (to check if the first one is bigger)"
[ ${number1%.*} -eq ${number2%.*} ] && [ ${number1#*.} \> ${number2#*.} ] || [ ${number1%.*} -gt ${number2%.*} ];
result=$?
if [ "$result" -eq 0 ]; then result=1; else result=0; fi
echo "... is_first_floating_number_bigger: result is: ${result}"
if [ "$result" -eq 0 ]; then
echo "... is_first_floating_number_bigger: ${number1} is not bigger than ${number2}"
else
echo "... is_first_floating_number_bigger: ${number1} is bigger than ${number2}"
fi
__FUNCTION_RETURN="${result}"
}
只需要将函数保存在一个单独的.sh文件中,并像这样包含它:
. /path/to/the/new-file.sh
AWK和类似的工具(我在盯着你看sed…)应该被扔进旧项目的垃圾箱里,因为它们的代码是用永不读的语言写的,所以每个人都不敢碰。
或者你是一个相对罕见的需要优先考虑CPU使用优化而不是代码维护优化的项目……既然如此,那就继续吧。
如果不是,那么就使用一些可读且显式的语言,比如Python。你的程序员同事和未来的自己都会感谢你。您可以像所有其他代码一样,在Bash中使用Python代码。
num1=3.17648E-22
num2=1.5
if python -c "exit(0 if $num1 < $num2 else 1)"; then
echo "yes, $num1 < $num2"
else
echo "no, $num1 >= $num2"
fi
更加方便
这可以使用Bash的数值上下文更方便地完成:
if (( $(echo "$num1 > $num2" |bc -l) )); then
…
fi
解释
通过基本计算器命令bc的管道将返回1或0。
选项-l等价于——mathlib;它加载标准数学库。
将整个表达式括在双括号(())之间会将这些值分别转换为true或false。
请确保安装了bc基本计算器包。
注意:指数符号应该写成*10^;不是E,也不是E。
例如:
$ echo "1*10^3==1000" |bc
1
而
$ echo "1E3==1000" |bc
0
这里讨论了克服这种bc限制的策略。