Kurangi dua variabel dalam Bash

220

Saya memiliki skrip di bawah ini untuk mengurangi jumlah file antara dua direktori tetapi COUNT=ekspresi tidak berfungsi. Apa sintaks yang benar?

#!/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
sumber
1
kemungkinan duplikat dari Bagaimana saya bisa menambahkan angka dalam skrip bash
Sorin

Jawaban:

224

Anda hanya perlu sedikit spasi tambahan di sekitar tanda minus, dan backticks:

COUNT=`expr $FIRSTV - $SECONDV`

Waspadai status keluar:

Status keluar adalah 0 jika EKSPRESI bukan nol atau 0, 1 jika EKSPRESI nol atau 0 .

Ingatlah ini ketika menggunakan ekspresi dalam skrip bash dalam kombinasi dengan set -e yang akan segera keluar jika perintah keluar dengan status bukan-nol.

Aaron McDaid
sumber
2
Jawaban ini juga berfungsi di shshell posix . Untuk portabilitas, Anda mungkin ingin menggunakan jawaban ini.
dinkelk
Perlu dicatat bahwa menurut Shellcheck, expr adalah kode kode karena ia kuno dan sulit digunakan: github.com/koalaman/shellcheck/wiki/SC2003
John Hamelink
369

Coba sintaks Bash ini alih-alih mencoba menggunakan program eksternal expr:

count=$((FIRSTV-SECONDV))

BTW, sintaks yang benar menggunakan expradalah:

count=$(expr $FIRSTV - $SECONDV)

Namun perlu diingat penggunaannya exprakan lebih lambat daripada sintaks Bash internal yang saya berikan di atas.

anubhava
sumber
4
Formulir ini lebih cepat daripada menggunakan program eksternal expr.
nsg
Ini berfungsi tanpa backticks, tetapi boleh saya tahu mengapa? +1 untuk answe.r
Amal Murali
2
Terima kasih. Backtick adalah sintaks shell lama. BASH mendukung $(command)sintaks baru untuk substitusi perintah. Juga karena BASH mendukung operasi aritmatika di $(( ... ))dalamnya lebih baik untuk tidak menggunakan utilitas eksternalexpr
anubhava
1
Saya tidak pernah baru Anda bisa referensi variabel tanpa "$", sangat menarik. Ini berfungsi di Ubuntu 12,14 hanya FYI.
Madatter
1
@ AlikElzin-kilaka: Dalam bash $(( ... ))digunakan untuk mengevaluasi ekspresi aritmatika.
anubhava
30

Kamu bisa memakai:

((count = FIRSTV - SECONDV))

untuk menghindari proses terpisah, sesuai transkrip berikut:

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

Ruang putih itu penting, exprmengharapkan operan dan operatornya sebagai argumen terpisah. Anda juga harus menangkap output. Seperti ini:

COUNT=$(expr $FIRSTV - $SECONDV)

tetapi lebih umum untuk menggunakan ekspansi aritmatika bawaan:

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

Beginilah cara saya selalu mengerjakan matematika di Bash:

count=$(echo "$FIRSTV - $SECONDV"|bc)
echo $count
Pureferret
sumber
5
itu hanya perlu jika Anda berurusan dengan angka floating point.
glenn jackman
2
Saya menyadari itu, tapi saya lebih suka membiasakan menangkap kasus-kasus itu dengan |bcperintah ketik daripada melewatkannya sekali atau dua kali. Stroke berbeda untuk orang yang berbeda seperti yang mereka katakan.
Pureferret
5

Untuk aritmatika integer sederhana, Anda juga dapat menggunakan perintah builtin let .

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

Untuk info lebih lanjut let, lihat di sini .

Shawn Chin
sumber
@ another.anon.coward Tautan Anda lebih baik daripada saya +1. (... dan mencuri tautan)
Shawn Chin
Punya banyak masalah dalam mendapatkan ini berfungsi. Akhirnya ini berhasil - let "sanity_check_duration=sanity_check_duration_end_time_delay_sec - sanity_check_duration_start_time_delay_sec"(menghilangkan tanda dolar dari variabel)
Sandeepan Nath
2

Atau dengan 3 metode yang disarankan Anda dapat mencoba letyang melakukan operasi aritmatika pada variabel sebagai berikut:

let COUNT=$FIRSTV-$SECONDV

atau

let COUNT=FIRSTV-SECONDV

another.anon.coward
sumber
0

Gunakan 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

Keluaran

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