【发布时间】:2011-02-27 20:54:50
【问题描述】:
我有 Python 脚本 bgservice.py,我希望它一直运行,因为它是我构建的 Web 服务的一部分。即使在我注销 SSH 后,如何让它继续运行?
【问题讨论】:
我有 Python 脚本 bgservice.py,我希望它一直运行,因为它是我构建的 Web 服务的一部分。即使在我注销 SSH 后,如何让它继续运行?
【问题讨论】:
替代答案:tmux
tmux
tmux 中启动你想要的进程,例如python3 main.py
Ctrl+b 离开tmux 会话,然后d
现在可以安全地退出远程计算机。当您回来时使用tmux attach 重新进入tmux 会话。
如果您想启动多个会话,请使用Ctrl+b 命名每个会话,然后使用$。然后输入您的会话名称。
列出所有会话使用tmux list-sessions
要附加一个正在运行的会话,请使用tmux attach-session -t <session-name>。
【讨论】:
首先,您需要在 Python 脚本中添加 shebang 行,如下所示:
#!/usr/bin/env python3
如果您安装了多个 Python 版本,则此路径是必需的,/usr/bin/env 将确保采用您的 $$PATH 环境变量中的第一个 Python 解释器。你也可以硬编码你的 Python 解释器的路径(例如#!/usr/bin/python3),但这不灵活,在其他机器上也不能移植。接下来,您需要设置文件的权限以允许执行:
chmod +x test.py
现在您可以使用忽略挂断信号的nohup 运行脚本。这意味着您可以在不停止执行的情况下关闭终端。另外,不要忘记添加&,以便脚本在后台运行:
nohup /path/to/test.py &
如果您没有在文件中添加shebang,则可以使用以下命令运行脚本:
nohup python /path/to/test.py &
输出将保存在nohup.out 文件中,除非您像这里这样指定输出文件:
nohup /path/to/test.py > output.log &
nohup python /path/to/test.py > output.log &
如果您将命令的输出重定向到其他地方 - 包括 /dev/null - 那就是它的位置。
# doesn't create nohup.out
nohup command >/dev/null 2>&1
如果您使用的是nohup,这可能意味着您想在后台运行命令,方法是在整个过程的末尾加上另一个&:
# runs in background, still doesn't create nohup.out
nohup command >/dev/null 2>&1 &
您可以使用以下命令找到process 及其process ID:
ps ax | grep test.py
# or
# list of running processes Python
ps -fA | grep python
ps 代表process status
如果你想停止执行,你可以用kill命令kill它:
kill PID
【讨论】:
试试这个:
nohup python -u <your file name>.py >> <your log file>.log &
你可以在screen中运行上面的命令,然后跳出屏幕。
现在您可以通过以下方式跟踪您的 Python 脚本日志:tail -f <your log file>.log
要杀死你的脚本,你可以使用 ps -aux 和 kill 命令。
【讨论】:
你也可以使用Yapdi:
基本用法:
import yapdi daemon = yapdi.Daemon() retcode = daemon.daemonize() # This would run in daemon mode; output is not visible if retcode == yapdi.OPERATION_SUCCESSFUL: print('Hello Daemon')
【讨论】:
这是一个在 python 中使用装饰器的简单解决方案:
import os, time
def daemon(func):
def wrapper(*args, **kwargs):
if os.fork(): return
func(*args, **kwargs)
os._exit(os.EX_OK)
return wrapper
@daemon
def my_func(count=10):
for i in range(0,count):
print('parent pid: %d' % os.getppid())
time.sleep(1)
my_func(count=10)
#still in parent thread
time.sleep(2)
#after 2 seconds the function my_func lives on is own
您当然可以将bgservice.py 文件的内容替换为my_func。
【讨论】:
您可能会考虑将您的 python 脚本转换为适当的 python 守护程序,如 here 所述。
python-daemon 是一个很好的工具,可用于将 python 脚本作为后台守护进程运行,而不是永久运行的脚本。您将需要稍微修改现有代码,但其简单明了。
如果您在使用 python-daemon 时遇到问题,还有另一个实用程序 supervisor 可以为您做同样的事情,但在这种情况下,您不必编写任何代码(或修改现有代码),因为这是不合时宜的用于守护进程的盒子解决方案。
【讨论】:
如果您需要的是无论您是否登录,该进程都应该永远运行,请考虑将该进程作为守护进程运行。
supervisord 是一个很好的开箱即用的解决方案,可用于守护任何进程。它有另一个控制实用程序supervisorctl,可用于监视由主管运行的进程。
您无需编写任何额外代码或修改现有脚本即可完成这项工作。此外,详细的文档使这个过程更加简单。
在 python-daemon 上摸索了几个小时后,supervisor 是在几分钟内为我工作的解决方案。
希望这有助于尝试使 python-daemon 工作的人
【讨论】:
你也可以使用GNU screen,这是几乎每个 Linux/Unix 系统都应该有的。
如果您使用的是 Ubuntu/Debian,它的增强变体 byobu 也相当不错。
【讨论】:
如果你已经启动了进程,不想在nohup下杀死它重新启动,可以将它发送到后台,然后disown它。
Ctrl+Z(暂停进程)
bg(后台重启进程
disown %1(假设这是作业#1,使用jobs来确定)
【讨论】:
zsh shell 有一个选项可以让所有后台进程使用 nohup 运行。
在~/.zshrc 中添加以下行:
setopt nocheckjobs #don't warn about bg processes on exit
setopt nohup #don't kill bg processes on exit
那么你只需要像这样运行一个进程:python bgservice.py &,你就不再需要使用 nohup 命令了。
我知道没有多少人使用 zsh,但我会推荐它是一个非常酷的 shell。
【讨论】:
你可以不用,但我更喜欢screen。
【讨论】:
运行nohup python bgservice.py & 让脚本忽略挂断信号并继续运行。输出将放在nohup.out。
理想情况下,您可以使用supervise 之类的内容运行脚本,以便在(何时)死亡时重新启动它。
【讨论】:
nohup: ignoring in put and appending output to nohup.out'`,当我按下 enter 时,进程以状态 1 退出。这是怎么回事?
somecommand & 之后,在某些shell(bash、zsh 等)中,它会像[1] 12345 一样打印pid。否则你可以使用$!。