Linux Shell 进制错误 - value too great for base (数值太大不可为算数进制的基)

2011-06-24 15:28:54 旧日重来

Linux Shell 编程进行数学运算时,如果有字符 '0' 打头的数 Bash 会当做八进制解释,而这经常会引起问题。比如我们用"08"参加运算,本意是当做10进制的8,实际却会运行报错:value too great for base (数值太大不可为算数进制的基)。

通常情况下,直接写的数值不会以0打头,但如果是以前面命令的格式化输出为参数时,则往往不能控制。更危险的是,如果参与运算的以'0'打头数值达到了两位数或更多,且数值的每一位均在0-7之间,则 Shell 不会报错,但会输出错误的运算结果。下面是两个会出错的例子:

time=`date +%S`
time2=$((time%15))

当当前时间的秒正好是8或9的时候便会出现类似

aulddays:~$ time=`date +%S` 
aulddays:~$ time2=$((time%15))
bash: 08: value too great for base (error token is "08")\or
bash: 09: value too great for base (error token is "09")
-bash: 08: 数值太大不可为算数进制的基 (错误符号是 "08")\or
-bash: 09: 数值太大不可为算数进制的基 (错误符号是 "09")

的报错。又如,使用 seq 时,为了格式工整,往往会在个位数前补'0'对齐:

for hour in `seq -f %02g 0 23`
do
	hour2=$(($hour+10))
done

之类,当循环到8时也会出错。

究其原因,就是shell将“08”当成了8进制数,而8进制数中合法的最大的数字是'7',因而出错了。要解决这个问题,必须让Shell强制将"08"当成10进制来解释,具体做法,已第一个例子来讲需要改写成:

time=`date +%S`
time2=$((10#time%15))

这样便可让 Bash 将"08"解释为十进制数。前面已经说过,当以'0'打头的数值位数有多位时且每一位都在0-7之间时,Bash 会以八进制解释且不会报错。如果发现脚本执行结果中的数值出现了诡异的问题时,一定要检查一下是否是整数进制发生了错误

查看:原文地址;来源:live.aulddays.com


注意:本站所有文章除特别说明外均为原创,版权所有,转载请务必以超链接方式注明作者出处,并禁止用作商业用途