我想要一个变量每秒增加 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"