Sottrai due variabili in Bash


220

Ho lo script qui sotto per sottrarre il numero di file tra due directory ma l' COUNT=espressione non funziona. Qual è la sintassi corretta?

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

Risposte:


224

Hai solo bisogno di un po 'di spazio in bianco attorno al segno meno e di backtick:

COUNT=`expr $FIRSTV - $SECONDV`

Essere consapevoli dello stato di uscita:

Lo stato di uscita è 0 se EXPRESSION non è né nullo né 0, 1 se EXPRESSION è nullo o 0 .

Tienilo a mente quando usi l'espressione in uno script bash in combinazione con set -e che uscirà immediatamente se un comando esce con uno stato diverso da zero.


2
Questa risposta funziona anche nella shshell posix . Per la portabilità, potresti voler usare questa risposta.
Dinkelk,

Vale la pena notare che secondo Shellcheck, expr è un codemell a causa del fatto che è antiquato e difficile da usare: github.com/koalaman/shellcheck/wiki/SC2003
John Hamelink

369

Prova questa sintassi di Bash invece di provare a utilizzare un programma esterno expr:

count=$((FIRSTV-SECONDV))

A proposito, la sintassi corretta dell'uso exprè:

count=$(expr $FIRSTV - $SECONDV)

Ma tieni presente che l'utilizzo exprsarà più lento della sintassi Bash interna che ho fornito sopra.


4
Questo modulo è molto più rapido rispetto all'utilizzo del programma esterno expr.
nsg

Funziona senza i backtick, ma posso sapere perché? +1 per answe.r
Amal Murali,

2
Grazie. Backtick è la vecchia sintassi della shell. BASH supporta la nuova $(command)sintassi per la sostituzione dei comandi. Anche dal momento che BASH supporta operazioni aritmetiche $(( ... )), è meglio non usare un'utilità esternaexpr
anubhava,

1
Non ho mai saputo che potresti fare riferimento a variabili senza "$", molto interessante. Questo funziona su Ubuntu 12,14 solo FYI.
MadHatter,

1
@ AlikElzin-kilaka: in bash $(( ... ))viene utilizzato per valutare le espressioni aritmetiche.
Anubhava,

30

Puoi usare:

((count = FIRSTV - SECONDV))

per evitare di invocare un processo separato, secondo la seguente trascrizione:

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

12

Lo spazio bianco è importante, si expraspetta che i suoi operandi e operatori siano argomenti separati. Devi anche catturare l'output. Come questo:

COUNT=$(expr $FIRSTV - $SECONDV)

ma è più comune usare l'espansione aritmetica integrata:

COUNT=$((FIRSTV - SECONDV))

12

È così che faccio sempre matematica in Bash:

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

5
questo è necessario solo se hai a che fare con numeri in virgola mobile.
Glenn Jackman,

2
Me ne rendo conto, ma preferirei prendere l'abitudine di catturare quei casi con un |bccomando type piuttosto che perderlo una o due volte. Colpi diversi per gente diversa come si suol dire.
Pureferret,

5

Per una semplice aritmetica dei numeri interi, puoi anche usare il comando let incorporato .

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

Per maggiori informazioni su let, guarda qui .


@ another.anon.coward Il tuo link è migliore del mio +1. (... e rubare link)
Shawn Chin,

Ho avuto molti problemi a farlo funzionare. Alla fine ha funzionato - let "sanity_check_duration=sanity_check_duration_end_time_delay_sec - sanity_check_duration_start_time_delay_sec"(rimuovendo il simbolo del dollaro dalle variabili)
Sandeepan Nath,

2

In alternativa ai 3 metodi suggeriti puoi provare letche esegue operazioni aritmetiche su variabili come segue:

let COUNT=$FIRSTV-$SECONDV

o

let COUNT=FIRSTV-SECONDV


0

Usa 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

Produzione

$ ./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
$
Utilizzando il nostro sito, riconosci di aver letto e compreso le nostre Informativa sui cookie e Informativa sulla privacy.
Licensed under cc by-sa 3.0 with attribution required.