• 让进程在后台可靠运行的几种方法


    如果只是临时有一个命令需要长时间运行,什么方法能最简便的保证它在后台稳定运行呢?

    我们知道,当用户注销(logout)或者网络断开时,终端会收到 HUP(hangup)信号从而关闭其所有子进程。因此,我们的解决办法就有两种途径:要么让进程忽略 HUP 信号,要么让进程运行在新的会话里从而成为不属于此终端的子进程。

    1. nohup
      --
      nohup 的用途就是让提交的命令忽略 hangup 信号。让我们先来看一下 nohup 的帮助信息:
    run a command immune to hangups, with output to a non-tty
    

    只需在要处理的命令前加上 nohup 即可,标准输出和标准错误缺省会被重定向到 nohup.out 文件中。一般我们可在结尾加上"&"来将命令同时放入后台运行,也可用">filename 2>&1"来更改缺省的重定向文件名。

    1. setsid
      --
      如果我们换个角度思考,如果我们的进程不属于接受 HUP 信号的终端的子进程,那么自然也就不会受到 HUP 信号的影响了。setsid 就能帮助我们做到这一点。让我们先来看一下 setsid 的帮助信息:
    run a program in a new session
    

    setsid 的使用也是非常方便的,也只需在要处理的命令前加上 setsid 即可。值得注意的是,实战中的进程 ID(PID)为31094,而它的父 ID(PPID)为1(即为 init 进程 ID),并不是当前终端的进程 ID。

    1. &
      --
      这里还有一个关于 subshell 的小技巧。我们知道,将一个或多个命名包含在“()”中就能让这些命令在子 shell 中运行中,从而扩展出很多有趣的功能,我们现在要讨论的就是其中之一。

    当我们将"&"也放入“()”内之后,我们就会发现所提交的作业并不在作业列表中,也就是说,是无法通过jobs来查看的。让我们来看看为什么这样就能躲过 HUP 信号的影响吧。

    (ping www.ibm.com &)
    ps -ef | grep ping
    killall ping
    

    新提交的进程的父 ID(PPID)为1(init 进程的 PID),并不是当前终端的进程 ID。因此并不属于当前终端的子进程,从而也就不会受到当前终端的 HUP 信号的影响了。

    我们已经知道,如果事先在命令前加上 nohup 或者 setsid 就可以避免 HUP 信号的影响。但是如果我们未加任何处理就已经提交了命令,该如何补救才能让它避免 HUP 信号的影响呢?

    这时想加 nohup 或者 setsid 已经为时已晚,只能通过作业调度和 disown 来解决这个问题了。让我们来看一下 disown 的帮助信息:

    disown [-ar] [-h] [jobspec ...]
    	Without options, each jobspec is  removed  from  the  table  of
    	active  jobs.   If  the -h option is given, each jobspec is not
    	removed from the table, but is marked so  that  SIGHUP  is  not
    	sent  to the job if the shell receives a SIGHUP.  If no jobspec
    	is present, and neither the -a nor the -r option  is  supplied,
    	the  current  job  is  used.  If no jobspec is supplied, the -a
    	option means to remove or mark all jobs; the -r option  without
    	a  jobspec  argument  restricts operation to running jobs.  The
    	return value is 0 unless a jobspec does  not  specify  a  valid
    	job.
    

    需要注意的是,当使用过 disown 之后,会将把目标作业从作业列表中移除,我们将不能再使用jobs来查看它,但是依然能够用ps -ef查找到它。

    但是还有一个问题,这种方法的操作对象是作业,如果我们在运行命令时在结尾加了"&"来使它成为一个作业并在后台运行,那么就万事大吉了,我们可以通过jobs命令来得到所有作业的列表。但是如果并没有把当前命令作为作业来运行,如何才能得到它的作业号呢?答案就是用 CTRL-z(按住Ctrl键的同时按住z键)了!

    CTRL-z 的用途就是将当前进程挂起(Suspend),然后我们就可以用jobs命令来查询它的作业号,再用bg jobspec来将它放入后台并继续运行。需要注意的是,如果挂起会影响当前进程的运行结果,请慎用此方法。

    screen

    NAME
           screen - screen manager with VT100/ANSI terminal emulation
    
    SYNOPSIS
           screen [ -options ] [ cmd [ args ] ]
           screen -r [[pid.]tty[.host]]
           screen -r sessionowner/[[pid.]tty[.host]]
    
    DESCRIPTION
           Screen  is  a  full-screen  window  manager  that multiplexes a physical terminal
           between several processes (typically interactive shells).  Each virtual  terminal
           provides  the functions of a DEC VT100 terminal and, in addition, several control
           functions from the ISO 6429 (ECMA 48, ANSI X3.64) and ISO  2022  standards  (e.g.
           insert/delete  line and support for multiple character sets).  There is a scroll‐
           back history buffer for each virtual terminal and a copy-and-paste mechanism that
           allows moving text regions between windows.
    

    推荐 tmux

  • 相关阅读:
    Rasp技术介绍与实现(一)
    青藤云安全细述:三大云安全工具(CASB、CSPM、CWPP)的使用场景
    CWPP产品市场演进
    Global CyberSecurity Landscape
    Scala学习之路 (五)Scala的关键字Lazy
    Scala学习之路 (四)Scala的数组、映射、元组、集合
    Scala学习之路 (三)Scala的基本使用
    Scala学习之路 (二)使用IDEA开发Scala
    Scala学习之路 (一)Scala的安装
    Azkaban学习之路 (三)Azkaban的使用
  • 原文地址:https://www.cnblogs.com/weaming/p/5098317.html
Copyright © 2020-2023  润新知