远程执行命令不成功的问题
当我们部署、维护多台主机的时候,通常需要通过程序远程执行一些命令,然而通常会发现在本地可正常执行的命令,远程调用时就会不正常。 这通常都是因为远程调用的运行方式为: 远程登录-》调用-》退出登录,在’退出‘时终端会收到 HUP(hangup)信号从而关闭其所有子进程(自然包含了我们’调用‘的命令)。因此,我们的解决办法就有两种途径:要么让进程忽略 HUP 信号,要么让进程运行在新的会话里从而成为不属于此终端的子进程。
1. nohup
顾名思义,nohup 的用途就是让提交的命令忽略 hangup 信号。
一般我们可在结尾加上"&"来将命令同时放入后台运行,也可用">filename 2>&1"
来更改缺省的重定向文件名。
nohup 示例
[root@test ~]# nohup ping www.baidu.com & [1] 3059 nohup: appending output to `nohup.out' [root@test ~]# ps -ef |grep 3059 root 3059 984 0 21:06 pts/3 00:00:00 ping www.baidu.com root 3067 984 0 21:06 pts/3 00:00:00 grep 3059 [root@test ~]#
2。setsid
setsid可以让进程不属于接受 HUP 信号的终端的子进程,那么自然也就不会受到 HUP 信号的影响了。
setsid用法:
setsid --help
Usage:
setsid [options] <program> [arguments ...]
Options:
-c, --ctty set the controlling terminal to the current one
-h, --help display this help and exit
-V, --version output version information and exit
For more details see setsid(1).
可见 setsid 的使用也是非常方便的,也只需在要处理的命令前加上 setsid 即可。
setsid 示例
[root@test ~]# setsid ping www.baidu.com [root@test ~]# ps -ef |grep www.baidu.com root 31094 1 0 07:28 ? 00:00:00 ping www.baidu.com root 31102 29217 0 07:29 pts/4 00:00:00 grep www.baidu.com
值得注意的是,上例中我们的进程 ID(PID)为31094,而它的父 ID(PPID)为1(即为 init 进程 ID),并不是当前终端的进程 ID。
3。&
将一个或多个命名包含在“()”中就能让这些命令在子 shell 中运行中,当我们将"&"也放入“()”内之后,我们就会发现所提交的作业并不在作业列表中,也就是说,是无法通过jobs
来查看的。所以这样也就能躲过 HUP 信号的影响了。
subshell 示例
[root@test ~]# (ping www.baidu.com &) [root@test ~]# ps -ef |grep www.baidu.com root 16270 1 0 14:13 pts/4 00:00:00 ping www.baidu.com root 16278 15362 0 14:13 pts/4 00:00:00 grep www.baidu.com
从上例中可以看出,新提交的进程的父 ID(PPID)为1(init 进程的 PID),并不是当前终端的进程 ID。因此并不属于当前终端的子进程,从而也就不会受到当前终端的 HUP 信号的影响了。