Відніміть дві змінні в 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

Відповіді:


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
Амаль Муралі

2
Дякую. Backtick - це старий синтаксис оболонки. BASH підтримує новий $(command)синтаксис для заміни команд. Крім того, оскільки BASH підтримує арифметичні операції в $(( ... ))ньому, краще не використовувати зовнішню утилітуexpr
anubhava

1
Ніколи не знаю, що ви могли б посилатися на змінні без "$", дуже цікаво. Це працює на Ubuntu 12,14 просто FYI.
MadHatter

1
@ AlikElzin-kilaka: In bash $(( ... ))використовується для оцінки арифметичних виразів.
anubhava

30

Ви можете використовувати:

((count = FIRSTV - SECONDV))

щоб не викликати окремий процес відповідно до наступної стенограми:

pax:~$ FIRSTV=7
pax:~$ SECONDV=2
pax:~$ ((count = FIRSTV - SECONDV))
pax:~$ echo $count
5

12

Білий простір важливий, він exprочікує , що його операнди та оператори будуть окремими аргументами. Ви також повинні захопити вихід. Подобається це:

COUNT=$(expr $FIRSTV - $SECONDV)

але частіше використовувати вбудоване арифметичне розширення:

COUNT=$((FIRSTV - SECONDV))

12

Ось як я завжди займаюся математикою на Bash:

count=$(echo "$FIRSTV - $SECONDV"|bc)
echo $count

5
це потрібно лише в тому випадку, якщо ви маєте справу з числами з плаваючою комою.
glenn jackman

2
Я це усвідомлюю, але я скоріше скористаюся звичкою ловити ці випадки |bcкомандною командою, ніж пропускати її раз чи двічі. Як кажуть, різні штрихи для різних людей.
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" (видалення знака долара зі змінних)
Sandeepan Nath,

2

В якості альтернативи запропонованим 3 методам ви можете спробувати letвиконати арифметичні операції над змінними наступним чином:

let COUNT=$FIRSTV-$SECONDV

або

let COUNT=FIRSTV-SECONDV


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
$
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.