• 《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都会运行这个。

  • 相关阅读:
    关于异步取消线程以及异步销毁锁的探讨
    pthread_mutex_init & 互斥锁pthread_mutex_t的使用(转)
    Qt设置全局的widget的stylesheet
    浅析pthread_cond_wait(转)
    575 Skew Binary
    HDU 1229 还是A+B
    10370
    10300
    UVA 10071 Problem B Back to High School Physics
    UVA 10055 Problem A Hashmat the brave warrior
  • 原文地址:https://www.cnblogs.com/xcywt/p/7922788.html
Copyright © 2020-2023  润新知