序言
相信各位读者对秒表都不陌生,智能手机上通常都有这样一款软件
有一天心血来潮,便想要“复刻”一个命令行版本的秒表程序——主要是想尝试一下新学会的、“原地更新”的技能,而不是一行接一行地输出。程序的运行效果如下
那么这是怎么做的呢?
实现思路及代码
如何获取流逝的时间长度?
要实现一个秒表,首先要知道从开始计时至今过了多久。在*nix系统中,表示时刻的事实标准是Epoch Time,在shell
脚本中要获取Epoch Time可以用date
命令。再用首尾时刻相减便得到了期间流逝的秒数了,示例代码如下
1 | begin_at=$(date '+%s') |
双圆括号是一种在shell
脚本中执行算术运算的语法,其它语法可以参见Math in Shell Scripts。
如何换算为时分秒?
有了interval
中存储的总秒数后,换算成时分秒便是轻而易举的事情,示例代码如下
1 | ((hours=${interval} / 3600)) |
如何输出形如hh:mm:ss
的格式?
hh:mm:ss
的意思是分别用两个十进制数字显示时分秒,并以冒号分隔它们。如果有任何一个单位的数值小于10,便用字符0
填充左侧的空白。按这个格式,凌晨1点2分3秒便会显示为01:02:03
。
要在命令行中打印字符串,最容易想到的便是echo
命令,只可惜它不能方便地实现填充字符0
的需求。
强人所难也不是不行,示例代码如下
1 | hours=1 |
更优雅的方法是用printf
命令来自动填充左侧的字符0
1 | printf "%02d:%02d:%02d" ${hours} ${minutes} ${seconds} |
printf
命令类似于C语言中的printf
函数——它也支持打印转义的字符,下文会提到。
如何覆盖已经打印的内容?
今年以来我在断断续续地看Build Your Own Text Editor,学习如何开发文本编辑器。在这本小册子的第三章中,作者讲述了如何使用终端的转义序列(escape sequence
)来控制屏幕上显示的东西——这正是秒表程序所需要的。
例如,在终端输出转义序列\x1b[2J
可以清空屏幕,效果如下
为了覆盖已经打印出来的时分秒,需要:
- 先将光标移动到行首;
- 再清除从光标开始到行末的内容。
查阅《VT100 User Guide》第三章可以知道
- 要把光标移动到行首可以用转义序列
\x1b[8D
。之所以是8,是因为按照hh:mm:ss
输出时分秒后光标距离行首8个身位; - 要清除光标到行末内容可以用转义序列
\x1b[0K
(实际上,将光标移到行首只需要使用回车(carriage return
)即可,但它被解释为开启新的一行了)。
更优雅的方法甚至连转义序列也不需要,只要用tput
命令即可,示例代码如下
1 | echo -n '11:22:33' |
关于cr
和el
,以及更多可以传给tput
命令的参数,可以参见terminfo
的man
文档。
如何每隔一秒钟输出一次?
这大概是整个程序中最简单的需求了
1 | while [ 1 -eq 1 ] |
完整的秒表实现
至此,完整的秒表程序就可以实现出来了
1 | !/bin/bash |
运行后的效果正如本文开头的GIF所示。
全文完。