ลบสองตัวแปรใน 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

คุณต้องการพื้นที่ว่างเพิ่มอีกเล็กน้อยรอบเครื่องหมายลบและ backticks:

COUNT=`expr $FIRSTV - $SECONDV`

ระวังสถานะทางออก:

สถานะออกเป็น 0 ถ้าแสดงออกไม่เป็นโมฆะหรือ 0, 1 ถ้าการแสดงออกเป็นโมฆะหรือ 0

โปรดระลึกไว้เสมอเมื่อใช้นิพจน์ในสคริปต์ทุบตีร่วมกับset -eซึ่งจะออกทันทีหากคำสั่งออกจากสถานะไม่เป็นศูนย์


2
คำตอบนี้ยังใช้งานได้ในshเชลล์posix เพื่อความสะดวกในการพกพาคุณอาจต้องการใช้คำตอบนี้
dinkelk

มันน่าสังเกตว่าตาม Shellcheck, expr เป็น codesmell เนื่องจากมันเก่าและใช้งานยาก: github.com/koalaman/shellcheck/wiki/SC2003
John Hamelink

369

ลองไวยากรณ์ Bash นี้แทนที่จะลองใช้โปรแกรมภายนอกexpr:

count=$((FIRSTV-SECONDV))

BTW รูปแบบการใช้ที่ถูกต้องexprคือ:

count=$(expr $FIRSTV - $SECONDV)

แต่โปรดทราบว่าการใช้exprจะช้ากว่าไวยากรณ์ Bash ภายในที่ฉันให้ไว้ข้างต้น


4
แบบฟอร์มนี้มีขนาดเร็วกว่าการใช้โปรแกรมภายนอก expr
nsg

สิ่งนี้ใช้ได้โดยไม่มี backticks แต่ฉันจะรู้ได้ไหมว่าเพราะอะไร +1 สำหรับ answere.r
Amal Murali

2
ขอบคุณ Backtick เป็นไวยากรณ์เชลล์เก่า BASH รองรับ$(command)ไวยากรณ์ใหม่สำหรับการทดแทนคำสั่ง นอกจากนี้เนื่องจาก BASH สนับสนุนการดำเนินการทางคณิตศาสตร์ใน$(( ... ))นั้นจึงเป็นการดีกว่าที่จะไม่ใช้ยูทิลิตีภายนอกexpr
anubhava

1
ฉันไม่เคยใหม่คุณสามารถอ้างอิงตัวแปรโดยไม่มี "$" น่าสนใจมาก สิ่งนี้ใช้ได้กับ Ubuntu 12,14 เพียงแค่ FYI
MadHatter

1
@ AlikElzin-kilaka: ใน 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
มีความจำเป็นเฉพาะในกรณีที่คุณจัดการกับตัวเลขจุดลอยตัว
เกล็นแจ็

2
ฉันรู้ว่า แต่ฉันต้องการสร้างนิสัยในการจับกรณีเหล่านั้นด้วย|bcคำสั่งประเภทมากกว่าที่จะพลาดครั้งหรือสองครั้ง จังหวะที่แตกต่างกันสำหรับคนที่แตกต่างกันตามที่พวกเขาพูด
Pureferret

5

สำหรับการคำนวณเลขจำนวนเต็มอย่างง่ายคุณสามารถใช้คำสั่งbuiltin let

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

สำหรับข้อมูลเพิ่มเติมเกี่ยวกับการletดูที่นี่


@ another.anon.coward ลิงก์ของคุณดีกว่า +1 ของฉัน (... และขโมยลิงก์)
Shawn Chin

มีปัญหามากในการทำงานนี้ ในที่สุดก็ใช้งานได้ - 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
$
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.