Вычтите две переменные в Bash

220

У меня есть сценарий ниже, чтобы вычесть количество файлов между двумя каталогами, но COUNT=выражение не работает. Какой правильный синтаксис?

#!/usr/bin/env bash

FIRSTV=`ls -1 | wc -l`
cd ..
SECONDV=`ls -1 | wc -l`
COUNT=expr $FIRSTV-$SECONDV  ## -> gives 'command not found' error
echo $COUNT
Toop
источник
1
Возможный дубликат Как я могу добавить числа в сценарии bash
Сорин

Ответы:

224

Вам просто нужно немного лишних пробелов вокруг знака минус и обратных галочек:

COUNT=`expr $FIRSTV - $SECONDV`

Будьте в курсе статуса выхода:

Статус выхода равен 0, если EXPRESSION не равен ни нулю, ни 0, 1, если EXPRESSION равен нулю, или 0 .

Помните об этом при использовании выражения в скрипте bash в сочетании с set -e, который немедленно завершится, если команда завершится с ненулевым статусом.

Аарон МакДейд
источник
2
Этот ответ также работает в shоболочке posix . Для переносимости, вы можете использовать этот ответ.
dinkelk
Стоит отметить, что, согласно Shellcheck, expr - это кодовая ячейка из-за того, что он устарел и сложен в использовании: github.com/koalaman/shellcheck/wiki/SC2003
John Hamelink
369

Попробуйте использовать синтаксис Bash вместо использования внешней программы expr:

count=$((FIRSTV-SECONDV))

Кстати, правильный синтаксис использования expr:

count=$(expr $FIRSTV - $SECONDV)

Но имейте в виду, что использование exprбудет медленнее, чем внутренний синтаксис Bash, который я предоставил выше.

анубхава
источник
4
Эта форма на порядок быстрее, чем использование внешней программы expr.
Nsg
Это работает без пробелов, но я могу знать, почему? +1 за answe.r
Амаль Murali
2
Спасибо. Backtick - это старый синтаксис оболочки. BASH поддерживает новый $(command)синтаксис для подстановки команд. Также, поскольку BASH поддерживает арифметические операции $(( ... )), лучше не использовать внешнюю утилитуexpr
anubhava
1
Я никогда не знал, что вы можете ссылаться на переменные без "$", очень интересно. Это работает на Ubuntu 12,14 просто к сведению.
MadHatter
1
@ AlikElzin-kilaka: В bash $(( ... ))используется для оценки арифметических выражений.
анубхава
30

Ты можешь использовать:

((count = FIRSTV - SECONDV))

чтобы избежать вызова отдельного процесса, согласно следующей расшифровке:

pax:~$ FIRSTV=7
pax:~$ SECONDV=2
pax:~$ ((count = FIRSTV - SECONDV))
pax:~$ echo $count
5
paxdiablo
источник
12

Пробел важен, exprего операнды и операторы ожидаются как отдельные аргументы. Вы также должны захватить вывод. Как это:

COUNT=$(expr $FIRSTV - $SECONDV)

но чаще используется встроенное арифметическое расширение:

COUNT=$((FIRSTV - SECONDV))
Каролий хорват
источник
12

Вот как я всегда делаю математику в Bash:

count=$(echo "$FIRSTV - $SECONDV"|bc)
echo $count
Pureferret
источник
5
это необходимо, только если вы имеете дело с числами с плавающей запятой.
Гленн Джекман
2
Я понимаю это, но я бы предпочел поймать эти случаи с помощью |bcкоманды type, чем пропустить ее один или два раза. Как говорится, разные штрихи для разных людей.
Pureferret
5

Для простой целочисленной арифметики вы также можете использовать встроенную команду let .

 ONE=1
 TWO=2
 let "THREE = $ONE + $TWO"
 echo $THREE
    3

Для получения дополнительной информации let, смотрите здесь .

Шон Чин
источник
@ another.anon.coward Ваша ссылка лучше моей +1. (... и кража ссылки)
Шон Чин
У меня было много проблем с этим. Наконец это сработало let "sanity_check_duration=sanity_check_duration_end_time_delay_sec - sanity_check_duration_start_time_delay_sec"(удаление знака доллара из переменных)
Сандипан Нат
2

В качестве альтернативы предлагаемым 3 методам вы можете попробовать letвыполнить арифметические операции над переменными следующим образом:

let COUNT=$FIRSTV-$SECONDV

или

let COUNT=FIRSTV-SECONDV

another.anon.coward
источник
0

Используйте Python:

#!/bin/bash
# home/victoria/test.sh

START=$(date +"%s")                                     ## seconds since Epoch
for i in $(seq 1 10)
do
  sleep 1.5
  END=$(date +"%s")                                     ## integer
  TIME=$((END - START))                                 ## integer
  AVG_TIME=$(python -c "print(float($TIME/$i))")        ## int to float
  printf 'i: %i | elapsed time: %0.1f sec | avg. time: %0.3f\n' $i $TIME $AVG_TIME
  ((i++))                                               ## increment $i
done

Вывод

$ ./test.sh 
i: 1 | elapsed time: 1.0 sec | avg. time: 1.000
i: 2 | elapsed time: 3.0 sec | avg. time: 1.500
i: 3 | elapsed time: 5.0 sec | avg. time: 1.667
i: 4 | elapsed time: 6.0 sec | avg. time: 1.500
i: 5 | elapsed time: 8.0 sec | avg. time: 1.600
i: 6 | elapsed time: 9.0 sec | avg. time: 1.500
i: 7 | elapsed time: 11.0 sec | avg. time: 1.571
i: 8 | elapsed time: 12.0 sec | avg. time: 1.500
i: 9 | elapsed time: 14.0 sec | avg. time: 1.556
i: 10 | elapsed time: 15.0 sec | avg. time: 1.500
$
Виктория Стюарт
источник