远程执行命令不成功的问题

  当我们部署、维护多台主机的时候,通常需要通过程序远程执行一些命令,然而通常会发现在本地可正常执行的命令,远程调用时就会不正常。 这通常都是因为远程调用的运行方式为: 远程登录-》调用-》退出登录,在’退出‘时终端会收到 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 信号的影响了。

 

posted @ 2015-11-18 18:43  UltraMemorandum  阅读(496)  评论(0编辑  收藏  举报