Linux 是通过信号来实现进程间通信的,所以我们编写脚本时,可以通过捕捉特定的信号来控制Shell脚本的操作。下面简单列举一下常见的Linux 信号
生成信号
Linux 允许用键盘组合键生成两种基本的Linux信号,
1.中断(terminating)进程
Ctrl C 生成SIGINT 信号,并发送给当前 Shell 中运行的所有进程。这是我们强行终止Shell 进程常用的方法。
2.暂停(stopping)进程
Ctrl Z 生成SIGTSTP 信号,并发送给当前Shell 中运行的所有进程。
捕获信号
脚本可以通过 ' trap '命令来捕获信号,命令格式为:
trap commands signal1 signal2...
从上图上可以看出,通过 trap 捕获了SIGINT 信号,并在捕获是打印消息。
输出结果为:
从输出结果看出,我在第5秒和第8秒后,通过组合键 Ctrl C,触发了 SIGINT 信号。
EXIT 信号
EXIT 信号在脚本退出时生成,也可以在脚本中通过 trap 来捕获。
删除信息捕获
捕获某个信号并执行逻辑后,希望解除信号捕获,trap 命令也提供了方法
trap -- signals #删除signal 的信号捕获。
后台模式运行脚本
我们在终端启动脚本,但有时脚本运行时间比较久,那么终端就无法处理其他交互,这种情况时,我们可以让脚本在后台运行,终端可以立即返回进行其他交互。
以后台模式运行脚本非常简单,只要在脚本启动命令后面加上 '&' 符号即可。例如:
./my_script &
非控制台下运行脚本
有时我们需要后台运行的脚本在终端退出后仍然执行,那么就需要一个命令 'nohup'。
nohup ./my_script.sh &
通过该命令启动脚本后,当终端发过来SIGHUP信号时,脚本会忽略该信号。因为nohup 会将脚本进程与终端解除关联,所以脚本进程也就不再和STDOUT & STDERR关联了,nohup会将脚本的STDOUT&STDERR 重定向到nohup.out文件中。
重启停止的Job
上面我们讲过,通过Ctrl Z可以停止运行的脚本任务,停止后我们可以选择kill掉,如果我们希望停止的任务继续进行,如何操作呢?通过 ‘bg’命令可以重启停止的任务。格式为
bg (job_id) #重启对应job_id的job
从上图看出,我在脚本 "t16.1"中做了循环处理,当运行到第2次循环时,我停止了脚本,通过 jobs命令查看到脚本处于停止状态,通过 bg 命令,重启Job, 脚本继续执行。此处 bg 没有填写 job id, 那么命令会重启默认job, 默认 Job 会在 job id 旁边标识 “ ”,此例中只包含一个默认job,所有只需要输入 bg 命令即可。
调整脚本运行优先级
Linux内核负责将CPU时间按照优先级分配给不同的进程,Linux中优先级从-20(最高优先级)到 19(最低优先级),默认以0 启动所有进程。通过 nice 命令可以调整优先级,格式为
nice -n pri_level ./my_script # pri_level为优先级级别
上图中以优先级为11启动dash 进程,可以看到,NI 列中 dash 优先级为11。
renice 命令: 调整已运行命令的优先级。格式为
renice -n pri_level -p PID #修改进程PID的优先级。
Schedule Job
at 命令可以指定Linux 何时运行脚本。格式为
at [-f filename] time
at 命令会将Job保存到/var/spool/at 中,at的守护进程 atd 会定时去文件中读取schedule job,并按照计划时间执行,从而实现linux 上的 Schedule Job。at 命令支持的时间格式比较丰富,主要包括;
查看所有计划任务命令:‘apt‘., 删除 schedule job:: at rm line_num, 举例:
cron 时间表
cron 是一种特殊的时间表示格式:
min hour dayofmonth month dayofweek command
cron 可以用来实现定时计划。举例cron:
- 15 10 * * * command 表示每天10:15分执行 command.
- 15 10 * * 1 command 表示每周一10:15分执行command