在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

Answers:


224

您只需要在减号和反引号周围留一些多余的空格:

COUNT=`expr $FIRSTV - $SECONDV`

注意退出状态:

如果EXPRESSION既不为null也不为0,退出状态为0;如果EXPRESSION为null或0,退出状态为1

在bash脚本中将表达式与set -e结合使用时,请记住这一点,如果命令以非零状态退出,则该表达式将立即退出。


2
这个答案也可以在posix shshell中使用。为了便于携带,您可能需要使用此答案。
dinkelk

值得注意的是,根据Shellcheck的说法,expr是一种代码,因为它过时且难以使用:github.com/koalaman/shellcheck/wiki/SC2003
John Hamelink

369

尝试使用以下Bash语法,而不是尝试使用外部程序expr

count=$((FIRSTV-SECONDV))

顺便说一句,using的正确语法expr是:

count=$(expr $FIRSTV - $SECONDV)

但是请记住,使用expr它的速度将比我上面提供的内部Bash语法慢。


4
这种形式比使用expr外部程序要快得多。
nsg

这没有反引号,但我可以知道为什么吗?+1为answe.r
阿马尔·穆拉利

2
谢谢。Backtick是旧的Shell语法。BASH支持新$(command)的命令替换语法。另外,由于BASH支持算术运算,$(( ... ))因此最好不要使用外部实用程序expr
anubhava 2014年

1
我从不陌生,您可以不用“ $”来引用变量,这很有趣。仅在FYI上即可在Ubuntu 12,14上运行。
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
仅在处理浮点数时才需要。
glenn jackman 2011年

2
我意识到这一点,但是我宁愿养成使用|bc类型命令来捕获这些情况的习惯,而不是一两次错过它。正如他们所说,针对不同人的不同笔触。
Pureferret 2011年

5

对于简单的整数算术,您还可以使用内置的let命令。

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

有关更多信息let,请单击此处


@ another.anon.coward您的链接比我的+1好。(...并窃取链接)
肖恩·

要使此工作很麻烦。最终,这项工作奏效了- 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
$
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.