Subtrair duas variáveis ​​no Bash

220

Eu tenho o script abaixo para subtrair as contagens de arquivos entre dois diretórios, mas a COUNT=expressão não funciona. Qual é a sintaxe correta?

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

Respostas:

224

Você só precisa de um espaço em branco extra ao redor do sinal de menos e backticks:

COUNT=`expr $FIRSTV - $SECONDV`

Esteja ciente do status de saída:

O status de saída é 0 se EXPRESSION não for nulo nem 0, 1 se EXPRESSION for nulo ou 0 .

Lembre-se disso ao usar a expressão em um script bash em combinação com set -e, que será encerrado imediatamente se um comando sair com um status diferente de zero.

Aaron McDaid
fonte
2
Esta resposta também funciona no shshell posix . Para portabilidade, convém usar esta resposta.
dinkelk
Vale ressaltar que, de acordo com Shellcheck, expr é um código de código devido a ser antiquado e difícil de usar: github.com/koalaman/shellcheck/wiki/SC2003
John Hamelink
369

Experimente esta sintaxe do Bash em vez de tentar usar um programa externo expr:

count=$((FIRSTV-SECONDV))

BTW, a sintaxe correta do uso expré:

count=$(expr $FIRSTV - $SECONDV)

Mas lembre-se de que o uso exprserá mais lento que a sintaxe interna do Bash que forneci acima.

anubhava
fonte
4
Este formulário é magnitudes mais rápido do que usar o programa externo expr.
NSG
Isso funciona sem os retalhos, mas posso saber por quê? +1 para o answe.r
Amal Murali
2
Obrigado. Backtick é uma sintaxe antiga do shell. O BASH suporta nova $(command)sintaxe para substituição de comandos. Também desde apoio BASH operações aritméticas em $(( ... ))não, é melhor usar um utilitário externoexpr
anubhava
1
Eu nunca novo, você poderia fazer referência a variáveis ​​sem o "$", muito interessante. Isso funciona no Ubuntu 12,14 apenas para sua informação.
precisa
1
@ AlikElzin-kilaka: No bash $(( ... ))é usado para avaliar expressões aritméticas.
Anubhava
30

Você pode usar:

((count = FIRSTV - SECONDV))

para evitar a invocação de um processo separado, conforme a seguinte transcrição:

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

O espaço em branco é importante, exprespera que seus operandos e operadores sejam argumentos separados. Você também precisa capturar a saída. Como isso:

COUNT=$(expr $FIRSTV - $SECONDV)

mas é mais comum usar a expansão aritmética integrada:

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

É assim que eu sempre faço matemática no Bash:

count=$(echo "$FIRSTV - $SECONDV"|bc)
echo $count
Pureferret
fonte
5
isso só é necessário se você estiver lidando com números de ponto flutuante.
perfil completo de Glenn Jackman
2
Sei disso, mas prefiro criar o hábito de capturar esses casos com um |bccomando de tipo do que perdê-lo uma ou duas vezes. Cursos diferentes para pessoas diferentes, como eles dizem.
Pureferret
5

Para aritmética inteira simples, você também pode usar o comando let incorporado .

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

Para mais informações let, veja aqui .

Shawn Chin
fonte
@ another.anon.coward Seu link é melhor que o meu +1. (... e roubar link)
Shawn Chin
Teve muitos problemas para fazer isso funcionar. Finalmente isso funcionou - let "sanity_check_duration=sanity_check_duration_end_time_delay_sec - sanity_check_duration_start_time_delay_sec"(sinal de remoção de dólar a partir de variáveis)
Sandeepan Nath
2

Como alternativa aos 3 métodos sugeridos, você pode tentar letexecutar operações aritméticas em variáveis ​​da seguinte maneira:

let COUNT=$FIRSTV-$SECONDV

ou

let COUNT=FIRSTV-SECONDV

another.anon.coward
fonte
0

Use 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

Resultado

$ ./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
fonte