我希望一个变量每秒增加 1。
#!/bin/bash
var=0
watch -n 1 echo "$((var++))"
输出(几秒钟后):
Every 1.0s: echo 0
0
它在屏幕上没有改变,但是当我输入echo "$var"
输出时是 1。为什么?
我希望一个变量每秒增加 1。
#!/bin/bash
var=0
watch -n 1 echo "$((var++))"
输出(几秒钟后):
Every 1.0s: echo 0
0
它在屏幕上没有改变,但是当我输入echo "$var"
输出时是 1。为什么?
用 来做到这一点并不容易
watch
,但有可能。我会在下面解释。但是,如果您只想要一个每秒递增一次的计数器,并且实际上不必
watch
出于任何其他原因使用,您可以简单地使用 shell 循环和 a 来解决这个问题sleep
:如果无法将您的
watch
命令转换成这样的循环,或者当然如果您对为什么它不能按照您尝试的方式工作感兴趣,请继续阅读。首先,您的第一个问题是
Bash 将首先扩展您的算术表达式,然后运行命令,因此它实际上运行
然后递增
var
一次,这就是为什么你1
之后会得到价值。现在你可以把它放在单引号而不是双引号中,但是根本不会发生扩展,它会
$((var++))
一直按字面意思输出。我们可以通过它eval
再次评估字符串。然后,此评估将每秒发生一次。因为默认
watch
使用sh
,我们需要使用不同的算术语法,因为它不支持++
:$(( var += 1))
- 请注意,这将首先增加变量然后返回增加的值,$((var++))
这与首先返回原始值然后增加变量不同。不幸的是,这将打印
1
在手表内,最终值var
将保持不变0
。为什么?被监视的命令在子 shell 中运行。子shell 有自己的、独立的环境和变量。
export var
您可以通过在 之前运行一次来将父 shell(您键入的那个)的变量导出到它的子 shell(其中运行监视命令的那些)watch
,但这只会给它们读取访问权限。对变量的每一次更改,比如增量,仍然只发生在子外壳内部,而不是传播到父外壳。如果您
export var
按照描述,观察的输出将始终是您设置的任何起始值 + 1,但变量的最终值将保持起始值。所以,我们了解到,实际上没有办法直接将变量从子 shell 传递给它的父 shell,所以 watch 命令内部发生的任何事情都不会影响父 shell 的变量和环境,也不会在调用相同的命令
watch
。我们需要某种不绑定到 shell 环境的临时存储,我们可以在其中读取和写入当前的计数器变量。例如,一个文件:
这将创建一个具有随机名称 in 的临时文件,
/tmp
在导出的 shell 变量中记住它的名称,$temp
并将您的起始值 0 写入它。现在在你的手表里,你必须从那个文件中读取你的变量,然后你可以增加它并做任何你想做的事情,但最后你必须把你改变的值写回文件:
这最终将使您的手表计数。但是,不要忘记您必须再次将文件中的最后一个值读取到父 shell 变量中,如果您在手表结束后仍想访问它,并且不要忘记再次删除临时文件: