linux系统下的脱机工作——nohup、screen

什么是工作管理

当我们登陆系统取得bash shell之后,在单一终端机下同时进行多个工作的行为管理。在进行工作管理的行为中,其实每个工作都是目前bash的子进程,即彼此之间是有相关性的。我们无法以job control的方式由tty1的环境去管理tty2的bash。

几个重要概念:

  • 这些工作所触发的进程必须来自于你shell的子进程(只管理自己的shell)

  • 前台:你可以控制与执行命令的这个环境成为前台(foreground)

  • 后台:可以自行运行的工作,你无法用ctrl+c来终止它,可以使用bg/fg调用该工作

  • 后台中执行的进程不能等待terminal/shell终止它,可以使用bg/fg调用该工作

job control的管理

后台的工作状态又可以分为运行中以及暂停状态

  • 直接将命令丢到后台运行的"&"

    [root@localhost ~]# find / -name hee.php &
    [1] 8048   <====== [job number] PID
    
    [1]+  完成                  find / -name hee.php
    
    # 将输出都重定向,以免让屏幕乱糟糟
    [root@localhost ~]# tar -zcpf etc.tar.gz /etc >/dev/null 2>&1 &
    [1] 8211
  • 将目前工作丢到后台中“暂停” : ctrl+z

    想个情况:如果我正在使用vim ,却发现我有个档案不知道放在哪里,需要到bash 环境下进行搜寻,此时是否要结束vim 呢?呵呵!当然不需要啊!只要暂时将vim 给他丢到背景当中等待即可。例如以下的案例:

    [root@study ~]# vim ~/.bashrc  
    #在vim的一般模式下,按下[ctrl]-z这两个按键 
    [1]+ Stopped vim ~/.bashrc 
    [root@study ~]#    <==顺利取得了前景的操控权! 
    [root@study ~]# find / -print  ....(输出省略).... 

此时萤幕会非常的忙碌!因为萤幕上会显示所有的档名。请按下[ctrl]-z 暂停

[2]+ Stopped find / -print


  在vim的一般模式下,按下[ctrl]及z这两个按键,萤幕上会出现[1] ,表示这是第一个工作,而那个+代表最近一个被丢进背景的工作,且目前在背景下预设会被取用的那个工作(与fg这个指令有关)!而那个Stopped则代表目前这个工作的状态。在预设的情况下,使用[ctrl]-z丢到背景当中的工作都是『暂停』的状态喔!

* 查看目前的后台工作状态:jobs

  ```shell
  [root@study ~]# jobs [-lrs] 
  选项与参数:
  -l :除了列出job number 与指令串之外,同时列出PID 的号码;
  -r :仅列出正在背景run 的工作;
  -s :仅列出正在背景当中暂停(stop) 的工作。

  范例一:观察目前的bash当中,所有的工作,与对应的PID 
  [root@study ~]# jobs -l
  [1]- 14566 Stopped vim ~/.bashrc
  [2]+ 14567 Stopped find / -print

其实+代表最近被放到背景的工作号码, -代表最近最后第二个被放置到背景中的工作号码。 而超过最后第三个以后的工作,就不会有+/-符号存在了!

  • 将后台工作拿到前台来处理:fg

    [root@study ~]# fg %jobnumber 
    选项与参数:
    %jobnumber :jobnumber 为工作号码(数字)。注意,那个% 是可有可无的!
    
    范例一:先以jobs观察工作,再将工作取出: 
    [root@study ~]# jobs -l
    [1]- 14566 Stopped vim ~/.bashrc
    [2]+ 14567 Stopped find / -print
    [root@study ~]# fg       <==预设取出那个+的工作,亦即[2]。立即按下[ctrl]-z 
    [root@study ~]# fg %1    <==直接规定取出的那个工作号码!再按下[ctrl]-z 
    [root@study ~]# jobs -l 
    [1] + 14566 Stopped vim ~/.bashrc
    [2]- 14567 Stopped find / -print
  • 将工作在后台下的暂停状态变为运行中:bg

    我们刚刚提到,那个[ctrl]-z 可以将目前的工作丢到背景底下去『暂停』, 那么如何让一个工作在背景底下『 Run 』呢?我们可以在底下这个案例当中来测试!注意喔!底下的测试要进行的快一点!^_^

    范例一:一执行find / -perm /7000 > /tmp/text.txt后,立刻丢到背景去暂停!
    [root@study ~]# find / -perm /7000 > /tmp/text.txt 
    #此时,请立刻按下[ctrl]-z暂停!
    [3]+ Stopped find / -perm /7000 > /tmp/text.txt
    
    范例二:让该工作在背景下进行,并且观察他!!
    [root@study ~]# jobs ; bg %3 ; jobs
    [1] Stopped vim ~/.bashrc
    [2]- Stopped find / -print
    [3]+   Stopped                  find / -perm /7000 > /tmp/text.txt
    [3]+ find / -perm /7000 > /tmp/text.txt &
    [1]- Stopped vim ~/.bashrc
    [2]+ Stopped find / -print
    [3]    Running                  find / -perm /7000 > /tmp/text.txt &

    看到哪里有差异吗?呼呼!没错!就是那个状态列~以经由Stopping 变成了Running 啰!看到差异点,嘿嘿!指令列最后方多了一个& 的符号啰!代表该工作被启动在背景当中了啦!^_^

  • 管理后台当中的工作:kill

    刚刚我们可以让一个已经在背景当中的工作继续工作,也可以让该工作以fg 拿到前景来, 那么,如果想要将该工作直接移除呢?或者是将该工作重新启动呢?这个时候就得需要给予该工作一个讯号 (signal) ,让他知道该怎么作才好啊!此时, kill 这个指令就派上用场啦!

    [root@study ~]# kill -signal %jobnumber 
    [root@study ~]# kill -l 
    选项与参数:
    -l :这个是L 的小写,列出目前kill 能够使用的讯号(signal) 有哪些?
    signal :代表给予后面接的那个工作什么样的指示啰!用man 7 signal 可知:
    -1 :重新读取一次参数的设定档(类似reload);
    -2 :代表与由键盘输入[ctrl]-c 同样的动作;
    -9 :立刻强制删除一个工作;
    -15:以正常的程序方式终止一项工作。与-9 是不一样的。
    
    范例一:找出目前的bash环境下的背景工作,并将该工作『强制删除』。
    [root@study ~]# jobs
    [1]+ Stopped vim ~/.bashrc
    [2] Stopped find / -print
    [root@study ~]# kill -9 %2; jobs
    [1]+ Stopped vim ~/.bashrc
    [2]    Killed                   find / -print
    #再过几秒你再下达jobs一次,就会发现2号工作不见了!因为被移除了!
    
    范例二:找出目前的bash环境下的背景工作,并将该工作『正常终止』掉。
    [root@study ~]# jobs
    [1]+ Stopped vim ~/.bashrc
    [root@study ~]# kill -SIGTERM %1 
    # -SIGTERM与-15是一样的!您可以使用kill -l来查阅!
    # 不过在这个案例中, vim 的工作无法被结束喔!因为他无法透过kill 正常终止的意思!

脱机管理问题

nohup

要注意的是,我们在工作管理当中提到的『背景』指的是在终端机模式下可以避免[ctrl]-c中断的一个情境,你可以说那个是bash的背景,并不是放到系统的背景去喔!所以,工作管理的背景依旧与终端机有关啦!在这样的情况下,如果你是以远端连线方式连接到你的Linux主机,并且将工作以&的方式放到背景去,请问,在工作尚未结束的情况下你离线了,该工作还会继续进行吗?答案是『否』!不会继续进行,而是会被中断掉。

[root@study ~]# nohup [指令与参数]    <==在终端机前景中工作 
[root@study ~]# nohup [指令与参数] &  <==在终端机背景中工作

有够好简单的指令吧!上述指令需要注意的是, nohup 并不支援bash 内建的指令,因此你的指令必须要是外部指令才行。我们来尝试玩一下底下的任务吧!

# 1.先编辑一支会『睡着500秒』的程式: 
[root@study ~]# vim sleep500.sh
#!/bin/bash
/bin/sleep 500s
/bin/echo "I have slept 500 seconds."

# 2.丢到背景中去执行,并且立刻登出系统: 
[root@study ~]# chmod a+x sleep500.sh 
[root@study ~]# nohup ./sleep500.sh &
[2] 14812
[root@study ~]# nohup: ignoring input and appending output to `nohup.out' <==会告知这个讯息!
[root@study ~]# exit

如果你再次登入的话,再使用pstree 去查阅你的程序,会发现sleep500.sh 还在执行中喔!并不会被中断掉!这样了解意思了吗?由于我们的程式最后会输出一个讯息,但是nohup 与终端机其实无关了, 因此这个讯息的输出就会被导向『 ~/nohup.out 』,所以你才会看到上述指令中,当你输入nohup后, 会出现那个提示讯息啰。

如果你想要让在背景的工作在你登出后还能够继续的执行,那么使用nohup 搭配& 是不错的运作情境喔!可以参考看看!

screen

Screen是一款由GNU计划开发的用于命令行终端切换的自由软件。用户可以通过该软件同时连接多个本地或远程的命令行会话,并在其间自由切换。GNU Screen可以看作是窗口管理器的命令行界面版本。它提供了统一的管理多个会话的界面和相应的功能。

会话恢复

只要Screen本身没有终止,在其内部运行的会话都可以恢复。这一点对于远程登录的用户特别有用——即使网络连接中断,用户也不会失去对已经打开的命令行会话的控制。只要再次登录到主机上执行screen -r就可以恢复会话的运行。同样在暂时离开的时候,也可以执行分离命令detach,在保证里面的程序正常运行的情况下让Screen挂起(切换到后台)。这一点和图形界面下的VNC很相似。

多窗口

在Screen环境下,所有的会话都独立的运行,并拥有各自的编号、输入、输出和窗口缓存。用户可以通过快捷键在不同的窗口下切换,并可以自由的重定向各个窗口的输入和输出。Screen实现了基本的文本操作,如复制粘贴等;还提供了类似滚动条的功能,可以查看窗口状况的历史记录。窗口还可以被分区和命名,还可以监视后台窗口的活动。 会话共享 Screen可以让一个或多个用户从不同终端多次登录一个会话,并共享会话的所有特性(比如可以看到完全相同的输出)。它同时提供了窗口访问权限的机制,可以对窗口进行密码保护。

下面我们来演示screen的用法:

# 进入screen环境
# screen
# 然后执行下想要的命令
# tail -f /var/log/messages
# 暂时退出,做点别的事情
# 输入ctrl+a d
# 其他事做完了,想回来看看有没有新日志内容
# screen -ls
There is a screen on:
    6551.pts-0.iz8vb626ci0aehwsivxaydz  (Detached)
1 Socket in /var/run/screen/S-root.
# 然后我们通过会话id回到之前的环境
# screen -r 6551

# 看的差不多了,想退出screen
# exit

通过会话id感觉不是太方便,其实我们可以通过会话名来也行的

# screen -S wget
# wget http://soft.vpser.net/lnmp/lnmp1.7.tar.gz
# 这个时候使用了ctrl+a d或网络断线了或者终端被关闭了
# screen -r wget #通过该命令恢复
有时候screen异常退出可能会提示状态为Attached,可以执行:screen -D -r wget 进行恢复。