Linux如何在后台跑程序?是刚接触Linux的同学都会遇到的问题。大家使用Linux操作系统时,或许都遇到过以下类似场景。
- 场景1:我用xshell通过 telnet/ssh 远程登录服务器,跑脚本或执行一些耗时较长的任务,有时会因为网络不稳定或手贱等原因断开我的控制终端(如:xshell、sourceCRT)的远程连接状态,导致花费大量时间执行的程序又要重新跑;真难受!
- 场景2:我需要长时间稳定的跑脚本,希望能在Linux后台跑,关掉xshell也没事,且随时可以回来看输出信息或操作。
如果你曾被这类问题困扰过,或现在正为此发愁?那么请往下看。本文给大家分享我们在运维工作过程中最常用也最实用的两种后台跑程序方式。记得Mark
!
目录
应对手段
我们知道,当用户注销(logout)或者网络断开时,终端会收到 HUP(hangup)信号从而关闭其所有子进程(包括sshd及其子进程)。
因此,我们的解决办法就有两种途径(对应以下两种方式):
- 让进程忽略 HUP 信号,
- 让进程运行在新的会话里从而成为不属于此终端的子进程。
方式一:nohup 命令 &
nohup 是我们最常用的办法。顾名思义,nohup 就是让提交的命令忽略 hangup 信号。从而当ssh连接状态断开时,不会被系统中断掉。nohup 在使用上十分方便,只需在要处理的命令前加上 nohup 即可,标准输出和标准错误缺省会被重定向到 nohup.out 文件中。一般我们可在结尾加上 &
表示将命令放入后台运行。
比如我们测试在后台执行 ping www.baidu.com
命令,如下即可:
nohup ping www.baidu.com &
同时,会在输入命令的目录下生成一个nohup.log日志文件(如已有该文件则继续使用)
当然,也可加上 >filename 2>&1
来自行更改日志输出的目录及文件filename
。如下:
nohup ping www.baidu.com > /home/myout.log 2>&1 &
方式二:screen
我们认为Screen是目前最实用的Linux后台运行工具之一。简单的说,screen 提供了 ANSI/VT100 的终端模拟器,使它能够在一个真实终端下运行多个全屏的伪终端,每个伪终端我们称之为一个session。
相当于我们可以用xshell打开多个连接(session),不用时可以将这些连接置为离线状态 - Detached
,离线状态下session中的程序会正常运行,也就是后台运行
,并随时可以恢复继续使用,恢复使用时为Attached
在线状态,说明有人正在使用。
1、如何使用screen
上图是screen的session列表,图中Attached
代表有其他终端正在使用这个session,Detached
说明session处于离线状态,
下面一起来看一下我们使用Screen时,常用的一些命令:
screen -S name -> 创建一个名为:name 的session(伪终端)
screen -R name -> 先试图恢复离线的session。若找不到离线的,即建立新的session(伪终端)
screen -ls -> 列出所有已经存在的session
ctrl + a + d -> 退出当前的session
screen -d name -> 将指定的session作业离线。
screen -d -r name -> 离线当前session,恢复并回到name这个session
kill (session进程号) -> 杀掉某个session进程
实际操作流程示例:
1、创建一个名称为“chh”的session。
# 创建方式一
screen -S chh
# 创建方式二
screen -R chh
2、创建后会自动进入session,进行相关操作。
3、通过ctrl + a + d
退出当前session,使其后台(离线)运行
4、screen -ls 查看session列表
5、通过screen -r chh
或screen -r 9043(进程号)
登录离线session
6、杀掉“chh”这个session
kill 9043(进程号)
2、screen 工具需安装
- 如果你使用yum软件包管理工具:yum install screen
- 如果你使用APT软件包管理工具:apt-get install screen
另:手动下载地址:http://ftp.gnu.org/gnu/screen/
3、screen 后台运行原理
我们可以通过查看进程树来理解screen是如何不受ssh断开影响的。
- 这是不用screen的普通运行方式及进程树
[root@pvcent107 ~]# ping www.baidu.com &
[1] 9499
[root@pvcent107 ~]# pstree -H 9499
init─┬─Xvnc
├─acpid
├─atd
├─2*[sendmail]
└─sshd─┬─sshd───sshd───bash───ping
└─sshd───sshd───bash───pstree
我们可以看出,未使用 screen 时我们所处的 bash 是 sshd 的子进程,当 ssh 断开连接时,HUP 信号自然会影响到它下面的所有子进程(包括我们新建立的 ping 进程),杀掉所有子进程。
- 再看看使用 screen 运行进程的进程树
[root@pvcent107 ~]# screen -r chh
[root@pvcent107 ~]# ping www.baidu.com &
[1] 9488
[root@pvcent107 ~]# pstree -H 9488
init─┬─Xvnc
├─acpid
├─atd
├─screen───bash───ping
├─2*[sendmail]
如图可见,bash 是 screen 的子进程,而 screen 是 init(PID为1)的子进程。那么当 ssh 断开连接时,HUP 信号自然不会影响到 screen 下面的子进程了。
附、一张有故事的照片(十三)
最近这个图在我朋友圈刷屏了
是 LOL 的 s1 界面
看到时心中五味杂陈
那时,我还是个意气风发的网瘾少年
还有梦想
十年之后