我想要一个变量每秒递增 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 中运行。Subshell 有自己的、独立的环境和变量。
您可以通过在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] 删除。
我来说两句