Linux Shell 编程进行数学运算时,如果有字符 '0' 打头的数 Bash 会当做八进制解释,而这经常会引起问题。比如我们用"08"参加运算,本意是当做10进制的8,实际却会运行报错。
通常情况下,直接写的数值不会以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: 08: value too great for base (error token is "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 会以八进制解释且不会报错。如果发现脚本执行结果中的数值出现了诡异的问题时,一定要检查一下是否是整数进制发生了错误。
说明:
以0开头的数会被当做8进制来处理,以0X或者0x开头的数据会被当做16进制来处理。