01 事情起始
由于本人最近在学习shell,网上有很多shell编程的例子,今天看到以下这个:
依葫芦画瓢,我照着敲了一遍,结果发现执行起来是这样的,会无限循环输出#
图1
在网上咨询了一番,这个脚本主要存在问题为:
killall 是用来杀死指定名字的进程(kill processes by name),注意是根据名字来杀进程
$0:shell本身的文件名
所以文件中killall $0 是行不通的,killall $0 把脚本进程杀死,但是jindu & 吧jingdu函数放到后台去运行了,kill $0 的时候并不会退出后台运行的,所以虽然复制已完成,但是#还是死循环的输出,经过修改,脚本修正如下:
在执行,就恢复正常了
02 事后分析
这段原始脚本实际上有很多值得挖掘的地方,下面一一说来:
(一)while :
在shell流程控制中,while :和while true是一个意思,即条件为真即执行循环语句内容
(二)echo -n
echo -n :不换行输出,如下面这个例子:
echo -n "123"
echo "456"
最后结果是123456,而不是
123
456
拓展以下,还有一个常用的echo -e,用来处理特殊字符
若字符串中出现以下字符,则特别加以处理,而不会将它当成一般文字输出:
\a 发出警告声;
\b 删除前一个字符;
\c 最后不加上换行符号;
\f 换行但光标仍旧停留在原来的位置;
\n 换行且光标移至行首;
\r 光标移至行首,但不换行;
\t 插入tab;
\v 与\f相同;
\ 插入\字符;
\nnn 插入nnn(八进制)所代表的ASCII字符
(三 ) sleep 命令
Linux sleep命令可以用来将目前动作延迟一段时间,也可以理解为睡眠或者是休眠
sleep 1 睡眠1秒(没带单位默认是秒)
sleep 1s 睡眠1秒
sleep 1m 睡眠1分
sleep 1h 睡眠1小时
(四)函数后面加&
意思是将函数移动至后台进行
(五) cp -a
用于复制目录,连同目录的属性也会一并复制过去
(六) $0
$0 :shell文件本身名
这里拓展以下:
$$ Shell本身的PID(ProcessID)
$! Shell最后运行的后台Process的PID
$? 最后运行的命令的结束代码(返回值)
$- 使用Set命令设定的Flag一览
$* 所有参数列表。如"$*"用「"」括起来的情况、以"$1 $2 … $n"的形式输出所有参数。
$@ 所有参数列表。如"$@"用「"」括起来的情况、以"$1" "$2" … "$n" 的形式输出所有参数。
$# 添加到Shell的参数个数
$0 Shell本身的文件名
$1~$n 添加到Shell的各参数值。$1是第1参数、$2是第2参数…。
(七)killall
killall 是用来杀死指定名字的进程(kill processes by name),注意是根据名字来杀进程
killall kill pkill 的用法会另外开文章记录
$0:shell本身的文件名
03 事后总结
这个是一个很简单的例子,由于本人基础较差所以难以发现问题,有关shell实现进度条,在微信群里有位大神给了这样一段建议
进度条的常规做法是把慢命令放后台同时返回PID,在循环里用kill -0判断命令是否结束,否则打印特殊字符,截图博客里的代码基本上看到jindu就可以不用看了
如果只是想在cp文件的时候显示进度条,用rsync就行了
这里就又涉及两个知识点了,记录下学习去