《Linux命令行与shell脚本编程大全》第十六章 控制脚本

一些控制脚本的方式:向脚本发送信号、修改脚本优先级,在脚本运行时切换到运行模式 

16.1 处理信号

linux利用信号与运行在系统中的进程进行通信。

也可以通过对脚本进行编程,使其在收到特定信号时执行某些命令。从而控制脚本的操作。

16.1.1 重温Linux信号

比如下面这些常见的:

信号   值         描述

1    SIGUP     挂起进程

 2    SIGINT     终止进程

 3    SIGOUT    停止进程

 9    SIGKILL    无条件终止进程

15    SIGTERM   尽可能终止进程

……

 

默认情况下,bash shell会忽略收到的任何SIGOUT(3)和SIGTERM(15)。

但是会处理收到的SIGHUP(1)和SIGINT(2)。

shell会将这些信号传给shell脚本程序来处理,而shell脚本的默认行为是忽略这些信号。

可以在脚本中加入识别信号的代码,并执行命令来处理信号。

 

16.1.2 生成信号

bash shell允许用键盘上的组合键生成两种进本的linux信号。

1.中断进程

Ctrl+c会生成SIGINT信号。会发送给当前在shell中运行的所有进程。

2.暂停进程

在进程运行时暂停进程,无需终止它。让程序继续保留在内存中,并能从上次停止的位置继续运行。

有时打开了一个关键的系统文件锁,这就比较危险了。

但是这样可以在不终止进程的情况下能深入脚本内部一窥究竟。

Ctrl+z 会生成一个SIGTSTP信号,停止shell中运行的任何进程。

步骤:

$sleep 30

按下Ctrl + Z

$ls -l --forest

S列(进程状态)中,停止进程的状态为T。

$exit

这样会退出。

 

16.1.3 捕获信号

也可以不忽略信号,在信号出现时捕获它们并执行相应命令。

trap命令允许你来指定脚本要监看并从shell中拦截的linux信号。

如果脚本收到了trap中列出的信号,该信号不再由shell处理,而是交由本地处理。

 

命令格式: trap commands signals

 

16.1.4 捕获脚本退出

在脚本退出时进行捕获。

在trap命令后加上EXIT信号就行了。

按下Ctrl+C 和 自己运行退出都能被捕获到。

 

例子:

  1 #!/bin/bash

  2 trap " echo 'GoodBye everyone!!!'" EXIT # 捕获脚本退出时的信号

  3 # trap " echo 'sorry! I have trapped Ctrl+C'" SIGINT # 对应16.1.3 的例子。捕获Ctrl+C

  4 echo "This is Begin ......"

  5 count=1

  6 while [ $count -le 10 ]

  7 do

  8         echo "    count = $count"

  9         sleep 1

 10         count=$[ $count + 1 ]

 11 done

 12 echo "This is End!"

 

16.1.5 修改或移除捕获

在脚本中的不同位置进行不同的捕获处理,需要重新使用带有新选项的trap命令。

修改了信号捕获后,脚本处理信号的方式就会发生变化。

也可以删除以及设置好的捕获。在trap命令与希望恢复默认行为的信号列表之间加上两个破折号就行了。 

trap -- SIGINT

也可以用单破折号来恢复信号的默认行为。

例子:

  1 #!/bin/bash

  2 trap " echo 'I have trapped Ctrl+C'" SIGINT

  3 echo "This is Begin ......"

  4 count=1

  5 while [ $count -le 5 ]

  6 do

  7         echo "  First loop: $count"

  8         sleep 1

  9         count=$[ $count + 1 ]

 10 done

 11 #trap – SIGINT # 恢复默认行为

 12 #trap – SIGINT # 删除设置好的捕获

 13 trap " echo 'I modified the trap'" SIGINT # 在这个位置也捕获一下,以后捕获到就会走这里

 14 count=1

 15 while [ $count -le 5 ]

 16 do

 17         echo "    Second loop: $count"

 18         sleep 1

 19         count=$[ $count + 1 ]

 20 done

 21

 22 echo "This is End!"

 

 

16.2 以后台模式运行脚本

有的脚本可能要运行很长时间,你不想一直在命令行界面等着,这时你也没法做别的事情。这时候就需要后台运行脚本了。

 

ps命令可以看到很多进程都不是运行在终端显示器上的,这些就是后台进程。

在后台模式下,进程运行不会和终端会话上电STDIN STDOUT STDER关联。

 

16.2.1 后台运行脚本

命令后面加个取地址符就好了

例如:

$./test &

显示的第一行是shell分配给后台进程的作业号[]里面的。后面那个是进程的PID

后台进程结束时,会在终端显示一条消息表明已经结束了。

注意:后台进程运行时,任然会使用终端显示器来显示STDOUT和STDERR消息。

所以可以将后台运行的程序的STDOUT和STDERR进行重定向。

 

16.2.2 运行多个后台作业

同时启动多个后台作业就可以了。

在终端会话中使用后台进程要注意,ps命令的输出中,每一个后台进程都和终端会话(pts/0)终端联系在一起。如果终端会话退出,那么后台进程也会退出。

 

 

16.3 在非控制台下运行脚本

需求:在终端启动脚本,让脚本一直运行到结束,即使退出了终端会话。

nohup命令可以做到。它运行了另外一个命令来阻断所有发送给该进程的SIGHUP信号。这样终端退出时脚本也不会退出。

命令格式:

$nohup ./test &

nohup会自动将STDOUT和STDERR的消息重定向到一个名为nohup.out的文件中。

如果在同一个目录运行两次,第二次会追加到nohup.out。

 

16.4 作业控制

重启停止的进程需要向其发送一个SIGCONT信号。

启动、停止、终止、恢复作业这些功能统称为作业控制

16.4.1 查看作业

jobs命令允许查看shell当前正在处理的作业。

命令格式:

$jobs

 

用法:

$./test > log.txt &  # 后台运行一个作业,不能马上结束

$jobs  # 这里就可以看到了

还有一些其他的选项:

-l 列出PID和作业号   -r 只列出运行中的作业  -s 只列出停止的作业 ……….

例子:

  1 #!/bin/bash

  2 echo "Script process ID:$$"

  3 count=1

  4 while [ $count -le 10 ]

  5 do

  6         echo "count = $count"

  7         sleep 1

  8         count=$[ $count + 1 ]

  9 done

 10

 11 echo "This is end, Bye!!!"

运行多次:

 

带加号+的:当做默认作业(被当成作业控制命令的操作对象)

默认作业完成后,执行下一个作业(带减号-的)。任何时候都只有一个带加号和一个带减号的作业。

 

还可以这样:

用kill杀死当前默认作业。那么值钱带减号的就变成默认作业了。

16.4.2 重启停止的作业

可以将已停止的作业作为后台进程或前台进程(会接管你当前工作的终端)重启。

用bg命令实现。

bg 加上作业号。

不加作业号可以重启默认作业。

当有多个作业时必须加上作业号。

实例:

 

Ctrl + Z停止作业。

注意:bg 重启后是后台作业,ctrl + c 是接受不到的。

前台模式重启作业,可用带作业号的fg命令

比如:fg 2

例子就像上面一样,把bg换成fg就好了。

16.5 调整谦让度

在多任务操作系统中,内核负责将cpu时间分配给系统上运行的每一个进程。

调度优先级是内核分配给进程的CPU时间。

在linux系统中,由shell启动的所有进程的调度优先级默认都是相同的,

 

调度优先级是个整数值(-20 -- +19)。

-20是最高优先级,+19是最低优先级。

可以通过nice命令来提高或者降低优先级。

 

16.5.1 nice命令

可以设置启动时的调度优先级。

nice –n 来指定新的优先级别。

比如:

$nice –n 10 ./test > test.txt &

注意:必须将nice命令和要启动的命令放在同一行。

可以通过ps命令查看谦让度值(NI列)

$ps –p 3634 –o pid,ppid,ni,cmd   //  这里3634是进程对应的PID

 

还可以这样:省掉-n,在破折号后面跟上优先级就好了。

$nice -10 ./test > test.txt &

 

如果想提高优先级会失败,nice组织普通系统用户来提高命令的优先级。

 

16.5.2 renice命令

改变系统上已经运行的命令的优先级。可以通过renice实现。

比如:

$./test &

… 此时已经运行了,假设PID是3454

 

$renice –n 10 –p 3454  // 这样会自动更新当前运行进程的优先级。

注意:

只能对属于你的进程执行renice

只能通过renice降低进程的优先级

root用户可以通过renice来调度任意进程的优先级

 

 

16.6 定时运行作业

可以在某个预设时间运行脚本。

方法:at命令和cron表

 

16.6.1 用at命令来计划执行作业

at命令会将作业提交到队列中,指定shell何时运行该作业。at的守护进程atd会以后台模式运行,检查作业队列来运行作业。

atd会检查某个特殊目录(通常在/var/spool/at)来获取at命令提交的作业。默认情况下,atd会没60s检查一下这个目录,如果设置的运行时间和当前时间匹配,atd守护进程就会运行此作业。

这个只能指定时间运行,不能循环运行。

 

1.at命令的基本格式

at [-f filename] time

-f指定脚本名

time 指定了linux系统何时运行该作业。 at可以识别多种不同的时间格式。

 

使用at命令该作业会被提交到作业队列(job quene

针对不同的优先级,存在26种不同的作业队列,作业队列通常用小写字母a-z和A-Z来指代。

作业队列的字母排序越高,作业运行的优先级就越低(nice值越大)。

默认情况下at的作业会被提交到a作业队列。可以用-q参数指定不同的队列字母。

 

2.获取作业的输出

显示器不会关联到该作业,取而代之的是linux将提交该作业的用户的电子邮件地址作为STDOUT和STDERR。任何标准输出和标准错误都会通过邮件发送给用户。

如果没有关联电子邮件就无法获得输出,所以最好在脚本中对STDERR和STDOUT进行重定向

at的-M选项用来屏蔽作业产生的输出信息。

例子:就是重定向的例子。这里不写了。

 

3. 列出等待的作业

atq命令可以查看系统中有哪些作业在等待。

我的atq是自己装的,好像不会默认安装。

 

作业列表列出了作业号,系统运行该作业的日期以及所在的作业队列。上面的都在a队列。

 

4.删除作业

atrm删除等待中的作业,后面接作业号。

比如:

$atrm 3

 

16.6.2安排需要定期执行的脚本

可以设置每天指定时间运行一次,或者每周一次,每月一次。

cron程序可以安排定期执行的作业。cron程序会在后台运行并检查一个特殊的表,来获取已安排执行的作业。

1.cron时间表

采用一种特别的格式来指定作业何时运行。格式如下:

min hour dayofmounth month dayofweek command

dayofmounth:指定月份中的日期值(1-31

dayofweek:表示指定周的第几天

cron时间表允许你用特定值,取值范围(比如1~5)或者通配符*来指定条目。

比如:

每天10:15运行命令:15 10 * * * command

每周1的4:15运行:15 4 * * 1 command

每个月第一天12点:0 12 1 * * command

 

command需要指定要运行命令或脚本的全路径名,后面还可以接参数和重定向符号

 

2. 构建cron时间表

crontab –l 列出已有的cron时间表。默认情况下,用户的cron时间表并不存在。

要为cron时间表添加条目可以用-e选项。

$cron -e

如果要设置定时运行自己的程序,就需要添加条目了。

 

3.浏览cron目录

有4个预配置的cron脚本目录可以供我们使用。hourly,dialy,monthly,weekly。

查看:

$ls /etc/eron.*ly

假如脚本需要每天运行一次,将它复制到daily就可以了。其他的同理

 

4.anacron程序

cron程序最大的问题是假定linux系统是24小时一直开机的。除非是服务器,否则不一定会24小时一直在。

关机的时候就有可能会错过某些需要运行的作业。系统开机时cron程序不会运行那些错过的作业。anacron程序就是为了解决这个问题的。

如果anacron知道某个作业错过了执行时间,它会尽快运行该作业。anacron程序只会处理位于cron目录的程序,比如/etc/cron.monthly。

anacron不会处理执行时间需求小于一天的脚本

 

16.6.3 使用新的shell启动脚本

如果每次运行脚本的时候都能够启动一个新的bash shell,将会非常的方便。(这个我理解不来)

 

补充第六章的内容

用户登录bash shell需要运行的启动文件。

$HOME/.bash_profile

$HOME/.bash_login

$HOME/.profile

每次启动一个新shell时,bash shell都会运行.bashrc文件。

假如在.bashrc最后加echo “I am new shell”。这样每打开一个新的shell都会运行这个。

posted @ 2017-11-29 21:19  xcywt  阅读(271)  评论(0编辑  收藏  举报
作者:xcywt
出处:https://www.cnblogs.com/xcywt//
本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。
如果文中有什么错误,欢迎指出。以免更多的人被误导。