通过 watch 命令增加变量

通过 watch 命令增加变量

我想要一个变量每秒增加 1。

#!/bin/bash
var=0
watch -n 1 echo "$((var++))" 

输出(几秒后):

Every 1.0s: echo 0
0

屏幕上没有变化,但是我输入时echo "$var"输出是 1。为什么?

答案1

用 来实现这一点并不容易watch,但并非不可能。我将在下面进行解释。

但是,如果您只想要一个每秒增加一次的计数器,而实际上不必watch因为其他原因使用它,那么您可以简单地用一个 shell 循环和一个来解决这个问题sleep

var=0
while sleep 1 ; do echo "$((var++))" ; done

如果无法将您的watch命令转换为这样的循环,或者当然如果您对为什么它不能按照您尝试的方式工作感兴趣,请继续阅读。


首先,你的第一个问题是

watch -n 1 echo "$((var++))" 

Bash 将首先扩展你的算术表达式,然后运行该命令,因此它实际上运行

watch -n 1 echo 0

然后增加var一次,这就是为什么你1之后会得到值。


现在你可以用单引号代替双引号,但这样根本不会发生任何扩展,并且它会一直输出$((var++))。我们可以让它eval再次评估字符串。然后这个评估将每秒发生一次。

因为默认watch使用sh,我们需要使用不同的算术语法,因为它不支持++$(( var += 1))- 请注意,这将首先增加变量,然后返回增加后的值,$((var++))这与首先返回原始值然后增加变量不同。

watch -n 1 eval 'echo $((var += 1))'

不幸的是,这将打印1在手表内部,并且的最终值var将保持不变0

为何?监视的命令在子 shell 中运行。子 shell 有自己的、独立的环境和变量。


export var您可以在 之前运行一次,将父 shell(您键入的 shell)的变量导出到其子 shell(监视的命令在其中运行的 shell)watch,但这只会授予它们读取权限。对变量的每次更改(例如增量)仍然只发生在子 shell 中,而不会传播到父 shell。

如果您export var按照描述操作,所监视的输出将始终是您设置的起始值 + 1,但变量的最终值将保持为起始值。


因此,我们了解到实际上没有办法将变量从子 shell 直接传递给其父 shell,因此监视的命令内部发生的任何事情都不会影响父 shell 的变量和环境,更改也不能通过 在同一命令的调用中持续存在watch

我们需要某种不绑定到 shell 环境的临时存储,以便我们可以在其中读取和写入当前的计数器变量。例如,一个文件:

export temp=$(tempfile)
echo 0 > "$temp"

这将创建一个具有随机名称的临时文件/tmp,在导出的 shell 变量中记住它的名称$temp,并将您的起始值 0 写入其中。

现在在您的手表中,您必须从该文件中读取您的变量,然后您可以增加它并做任何您想做的事情,但最后您必须将更改的值写回文件:

watch -n 1 'read var < "$temp" ; echo "$((var+=1))" ; echo "$var" > "$temp"'

这最终会使你的 watch 计数。但是,不要忘记,如果你在 watch 结束后仍想访问它,你必须再次将文件中的最后一个值读入父 shell 变量,并且不要忘记再次删除临时文件:

read var < "$temp"
rm "$temp"

相关内容