我试图用var=$var+1
和var=($var+1)
来增加一个数字变量,但没有成功。该变量是一个数字,但bash似乎把它读成了一个字符串。
Bash版本4.2.45(1)-release (x86_64-pc-linux-gnu) 在Ubuntu 13.10上。
在bash中,有不止一种方法来增加一个变量,但你尝试的方法是不正确的。
你可以使用例如算术扩展。
var=$((var+1))
((var=var+1))
((var+=1))
((var++))
或者你可以使用let
。
let "var=var+1"
let "var+=1"
let "var++"
var=$((var + 1))
bash中的算术使用$((...))
语法。
感谢 [Radu Rădeanu'的回答][1],它提供了以下在 bash 中递增变量的方法。
var=$((var+1))
((var=var+1))
((var+=1))
((var++))
let "var=var+1"
let "var+=1"
let "var++"
也有其他的方法。 比如,看这个问题的其他答案。
let var++
var=$((var++))
((++var))
{
declare -i var
var=var+1
var+=1
}
{
i=0
i=$(expr $i + 1)
}
有这么多的选择,就会有这两个问题。
如果有,哪个性能最好?
#!/bin/bash
# To focus exclusively on the performance of each type of increment
# statement, we should exclude bash performing while loops from the
# performance measure. So, let's time individual scripts that
# increment $i in their own unique way.
# Declare i as an integer for tests 12 and 13.
echo > t12 'declare -i i; i=i+1'
echo > t13 'declare -i i; i+=1'
# Set i for test 14.
echo > t14 'i=0; i=$(expr $i + 1)'
x=100000
while ((x--)); do
echo >> t0 'i=$((i+1))'
echo >> t1 'i=$((i++))'
echo >> t2 '((i=i+1))'
echo >> t3 '((i+=1))'
echo >> t4 '((i++))'
echo >> t5 '((++i))'
echo >> t6 'let "i=i+1"'
echo >> t7 'let "i+=1"'
echo >> t8 'let "i++"'
echo >> t9 'let i=i+1'
echo >> t10 'let i+=1'
echo >> t11 'let i++'
echo >> t12 'i=i+1'
echo >> t13 'i+=1'
echo >> t14 'i=$(expr $i + 1)'
done
for script in t0 t1 t2 t3 t4 t5 t6 t7 t8 t9 t10 t11 t12 t13 t14; do
line1="$(head -1 "$script")"
printf "%-24s" "$line1"
{ time bash "$script"; } |& grep user
# Since stderr is being piped to grep above, this will confirm
# there are no errors from running the command:
eval "$line1"
rm "$script"
done
i=$((i+1)) user 0m0.992s
i=$((i++)) user 0m0.964s
((i=i+1)) user 0m0.760s
((i+=1)) user 0m0.700s
((i++)) user 0m0.644s
((++i)) user 0m0.556s
let "i=i+1" user 0m1.116s
let "i+=1" user 0m1.100s
let "i++" user 0m1.008s
let i=i+1 user 0m0.952s
let i+=1 user 0m1.040s
let i++ user 0m0.820s
declare -i i; i=i+1 user 0m0.528s
declare -i i; i+=1 user 0m0.492s
i=0; i=$(expr $i + 1) user 0m5.464s
当$i
被声明为整数时,似乎bash执行i+=1
最快。
let
语句似乎特别慢,而expr
是目前最慢的,因为它不是内置的。
如果你把$var
声明为一个整数,那么你第一次尝试的东西就会真正发挥作用。
$ declare -i var=5
$ echo $var
5
$ var=$var+1
$ echo $var
6
参考文献 变量类型,Bash初学者指南
所有答案中缺少一种方法--bc
。
$ VAR=7
$ bc <<< "$VAR+2"
9
$ echo $VAR
7
$ VAR=$( bc <<< "$VAR+1" )
$ echo $VAR
8
bc
是由[POSIX][1]标准指定的,所以应该存在于所有版本的Ubuntu和POSIX兼容系统中。
<<<
重定向可以改为echo "$VAR" | bc
以实现可移植性,但由于问题问的是bash
--只用<<<
就可以了。
[1]: http://pubs.opengroup.org/onlinepubs/9699919799/utilities/bc.html#tag_20_09
所有默认变量(let
、(())
等)都存在返回代码1
的问题。
这经常会引起麻烦,例如,在使用set -o errexit
的脚本中。
下面是我用来防止错误代码 "1 "从数学表达式中产生的错误代码 "0 "的方法。
math() { (( "$@" )) || true; }
math a = 10, b = 10
math a++, b+=2
math c = a + b
math mod = c % 20
echo $a $b $c $mod
#11 12 23 3
这一定是完成如此简单的任务的最糟糕的方式,但我只是想把它记录下来,我想是为了好玩(与代码高尔夫完全相反)。
$ var=0
$ echo $var
0
$ var="$(python -c 'print('$var'+1)')"
$ echo $var
1
或
$ var="$(printf '%s\n' $var'+1' | bc)"
$ echo $var
1
认真使用这里的其他更好的选择之一。