Subtrahieren Sie zwei Variablen in Bash

220

Ich habe das Skript unten, um die Anzahl der Dateien zwischen zwei Verzeichnissen zu subtrahieren, aber der COUNT=Ausdruck funktioniert nicht. Was ist die richtige Syntax?

#!/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
quelle
1
Mögliches Duplikat von Wie kann ich Zahlen in ein Bash-Skript
Sorin

Antworten:

224

Sie brauchen nur ein kleines Leerzeichen um das Minuszeichen und Backticks:

COUNT=`expr $FIRSTV - $SECONDV`

Beachten Sie den Exit-Status:

Der Exit-Status ist 0, wenn EXPRESSION weder null noch 0 ist, 1, wenn EXPRESSION null oder 0 ist .

Beachten Sie dies, wenn Sie den Ausdruck in einem Bash-Skript in Kombination mit set -e verwenden, das sofort beendet wird, wenn ein Befehl mit einem Status ungleich Null beendet wird.

Aaron McDaid
quelle
2
Diese Antwort funktioniert auch in der Posix- shShell. Aus Gründen der Portabilität möchten Sie möglicherweise diese Antwort verwenden.
Dinkelk
Es ist erwähnenswert, dass laut Shellcheck expr ein Codesmell ist, da es veraltet und schwer zu verwenden ist: github.com/koalaman/shellcheck/wiki/SC2003
John Hamelink
369

Versuchen Sie diese Bash-Syntax, anstatt ein externes Programm zu verwenden expr:

count=$((FIRSTV-SECONDV))

Übrigens exprlautet die korrekte Syntax :

count=$(expr $FIRSTV - $SECONDV)

Beachten Sie jedoch, dass die Verwendung exprlangsamer sein wird als die oben angegebene interne Bash-Syntax.

Anubhava
quelle
4
Diese Form ist um Größenordnungen schneller als die Verwendung des externen Programms expr.
nsg
Dies funktioniert ohne die Backticks, aber darf ich wissen warum? +1 für die answe.r
Amal Murali
2
Vielen Dank. Backtick ist eine alte Shell-Syntax. BASH unterstützt die neue $(command)Syntax für die Befehlssubstitution. Da BASH arithmetische Operationen unterstützt $(( ... )), ist es auch besser, kein externes Dienstprogramm zu verwendenexpr
anubhava
1
Ich nie neu, Sie könnten Variablen ohne das "$" referenzieren, sehr interessant. Dies funktioniert unter Ubuntu 12,14 nur zu Ihrer Information.
MadHatter
1
@ AlikElzin-kilaka: In bash $(( ... ))wird zur Auswertung von arithmetischen Ausdrücken verwendet.
Anubhava
30

Sie können verwenden:

((count = FIRSTV - SECONDV))

um zu vermeiden, dass ein separater Prozess gemäß dem folgenden Transkript aufgerufen wird:

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

Leerraum ist wichtig, exprerwartet seine Operanden und Operatoren als separate Argumente. Sie müssen auch die Ausgabe erfassen. So was:

COUNT=$(expr $FIRSTV - $SECONDV)

Es ist jedoch üblicher, die eingebaute arithmetische Erweiterung zu verwenden:

COUNT=$((FIRSTV - SECONDV))
Karoly Horvath
quelle
12

So mache ich in Bash immer Mathe:

count=$(echo "$FIRSTV - $SECONDV"|bc)
echo $count
Pureferret
quelle
5
Dies ist nur erforderlich, wenn Sie mit Gleitkommazahlen arbeiten.
Glenn Jackman
2
Mir ist das klar, aber ich würde es mir lieber zur Gewohnheit machen, diese Fälle mit einem |bcTypbefehl abzufangen, als ihn ein- oder zweimal zu verpassen. Verschiedene Schläge für verschiedene Leute, wie sie sagen.
Pureferret
5

Für eine einfache Ganzzahlarithmetik können Sie auch den eingebauten Befehl let verwenden .

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

Weitere Informationen letfinden Sie hier .

Shawn Chin
quelle
@ another.anon.coward Dein Link ist besser als meiner +1. (... und Link stehlen)
Shawn Chin
Hatte große Probleme damit. Schließlich funktionierte dies - let "sanity_check_duration=sanity_check_duration_end_time_delay_sec - sanity_check_duration_start_time_delay_sec"(Entfernen des Dollarzeichens aus Variablen)
Sandeepan Nath
2

Alternativ zu den vorgeschlagenen 3 Methoden können Sie versuchen let, arithmetische Operationen für Variablen wie folgt auszuführen:

let COUNT=$FIRSTV-$SECONDV

oder

let COUNT=FIRSTV-SECONDV

other.anon.coward
quelle
0

Verwenden Sie 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

Ausgabe

$ ./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
$
Victoria Stuart
quelle