Как увеличить переменную в Bash?

609

Я попытался увеличить числовую переменную, используя оба var=$var+1и var=($var+1)без успеха. Переменная - это число, хотя bash, похоже, читает ее как строку.

Выпуск версии 4.2.45 (1) для Bash (x86_64-pc-linux-gnu) в Ubuntu 13.10.

user221744
источник

Ответы:

948

Существует несколько способов увеличить переменную в bash, но то, что вы пробовали, неверно.

Вы можете использовать, например, арифметическое расширение :

var=$((var+1))
((var=var+1))
((var+=1))
((var++))

Или вы можете использовать let:

let "var=var+1"
let "var+=1"
let "var++"

Смотрите также: http://tldp.org/LDP/abs/html/dblparens.html .

Раду Рэдяну
источник
31
или ((++var))или ((var=var+1))или ((var+=1)).
gniourf_gniourf
6
Любопытно, что var=0; ((var++))возвращает код ошибки, пока var=0; ((var++)); ((var++))нет. Есть идеи почему?
phunehehe
15
@phunehehe Посмотри на help '(('. В последней строке написано:Returns 1 if EXPRESSION evaluates to 0; returns 0 otherwise.
Раду Радеану
2
Я подозреваю, что оценка нуля, 1поэтому совет @ gniourf_gniourf включает, ((++var))но не включает ((var++)).
DreadPirateShawn
4
это безопасно использовать let var++без кавычек?
wjandrea
161
var=$((var + 1))

Арифметика в Bash использует $((...))синтаксис.

Пол Танзини
источник
9
Значительно лучше, чем принятый ответ. Всего лишь на 10% больше места, вам удалось предоставить достаточное количество примеров (один из них достаточно - девять - это избыточно до того момента, когда вы просто хвастаетесь), и вы предоставили нам достаточно информации, чтобы знать, что ((...))является ключом к использованию арифметики. в баш. Я не осознавал, что просто смотрел на принятый ответ - я думал, что есть странный набор правил о порядке операций или что-то, что приводит к круглым скобкам в принятом ответе.
ArtOfWarfare
82

Анализ производительности различных вариантов

Благодаря ответу Раду Рэдяну, который предоставляет следующие способы увеличения переменной в 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)
}

Наличие множества вариантов приводит к этим двум вопросам:

  1. Есть ли разница в производительности между ними?
  2. Если так, то какой, который работает лучше всего?

Инкрементальный тест производительности кода:

#!/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

Заключение:

Кажется, bash работает быстрее всего, i+=1когда $iобъявлено как целое число. letутверждения кажутся особенно медленными, и expr, безусловно, самыми медленными, потому что они не являются встроенными.

wjandrea
источник
Очевидно, скорость коррелирует с длиной команды. Интересно, вызывают ли команды одни и те же функции.
MatthewRock
18

Там также это:

var=`expr $var + 1`

Обратите особое внимание на пробелы , а также ` не '

Хотя ответы Раду и комментарии являются исчерпывающими и очень полезными, они относятся к конкретным аспектам. Я знаю, что вы специально спрашивали о bash, но я подумал, что смогу перейти, так как я нашел этот вопрос, когда хотел сделать то же самое, используя sh в busybox под uCLinux. Это портативный за пределами Баш.

tphelican
источник
1
Вы также можете использоватьi=$((i+1))
wjandrea
Если $(...)в этой оболочке доступна подстановка процесса , я бы рекомендовал использовать ее вместо этого.
Радон Росборо
7

Во всех ответах отсутствует один метод - bc

$ VAR=7    
$ bc <<< "$VAR+2"
9
$ echo $VAR
7
$ VAR=$( bc <<< "$VAR+1" )
$ echo $VAR
8

bcопределяется стандартом POSIX , поэтому должен присутствовать во всех версиях Ubuntu и POSIX-совместимых систем. <<<Перенаправление может быть изменено , чтобы echo "$VAR" | bcпортативность, но поскольку вопрос касается bash- это нормально , просто использовать <<<.

Сергей Колодяжный
источник
6

Проблема с кодом возврата 1присутствует для всех вариантов по умолчанию ( let, (())и т. Д.). Это часто вызывает проблемы, например, в сценариях, которые используют 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
Юве
источник
0

Это должен быть худший способ выполнить такую ​​простую задачу, но я просто хотел документировать это для забавы, я думаю (полная противоположность кода гольфу).

$ var=0
$ echo $var
0
$ var="$(python -c 'print('$var'+1)')"
$ echo $var
1

или же

$ var="$(printf '%s\n' $var'+1' | bc)"
$ echo $var
1

Серьезно используйте один из других гораздо лучших вариантов здесь.

leetbacoon
источник