通过 watch 命令增加变量

咏叹调5

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

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

输出(几秒钟后):

Every 1.0s: echo 0
0

它在屏幕上没有改变,但是当我输入echo "$var"输出是 1.Why?

字节指挥官

用 来做到这一点并不容易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 中运行。Subshel​​l 有自己的、独立的环境和变量。


您可以通过在export var之前运行一次,将父 shell(您键入的那个)的变量导出到它的子 shell(在其中运行监视命令的那些)watch,但这只会给它们提供读访问权限。对变量的每次更改,如增量,仍然只发生在子 shell 内部,而不会传播到父 shell。

如果您export var如所描述的那样,监视的输出将始终是您设置的任何起始值 + 1,但变量的最终值将保持起始值。


所以,我们了解到实际上没有办法将变量从子 shell 直接传递给它的父 shell,所以在 watched 命令中发生的任何事情都不会影响父 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"'

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

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

本文收集自互联网,转载请注明来源。

如有侵权,请联系 [email protected] 删除。

编辑于
0

我来说两句

0 条评论
登录 后参与评论

相关文章