守护进程:通常被定义为一个后台进程,而且它不属于任何一个终端会话(terminal session)。许多系统服务由守护程序实施;如网络服务,打印等。
下面是转自一位网友写的编写守护进程的步骤:
1. 调用fork()以便父进程可以退出,这样就将控制权归还给运行你程序的
命令行或shell程序。需要这一步以便保证新进程不是一个进程组头领进程(process
group leader)。下一步,‘setsid()’,会因为你是进程组头领进程而失败。
2. 调用‘setsid()’ 以便成为一个进程组和会话组的头领进程。由于一个控制终端
与一个会话相关联,而且这个新会话还没有获得一个控制终端,我们的进程没
有控制终端,这对于守护程序来说是一件好事。
3. 再次调用‘fork()’所以父进程(会话组头领进程)可以退出。这意味着我们,一
个非会话组头领进程永远不能重新获得控制终端。
4. 调用‘chdir("/")’确认我们的进程不保持任何目录于使用状态。不做这个会导
致系统管理员不能卸装(umount)一个文件系统,因为它是我们的当前工作目录。
[类似的,我们可以改变当前目录至对于守护程序运行重要的文件所在目录]
5. 调用‘umask(0)’以便我们拥有对于我们写的任何东西的完全控制。我们不知
道我们继承了什么样的umask。
[这一步是可选的](译者注:这里指步骤5,因为守护程序不一定需要写文件)
6. 调用‘close()’关闭文件描述符0,1和2。这样我们释放了从父进程继承的标
准输入,标准输出,和标准错误输出。我们没办法知道这些文描述符符可能
已经被重定向去哪里。注意到许多守护程序使用‘sysconf()’来确认
‘_SC_OPEN_MAX’的限制。‘_SC_OPEN_MAX’告诉你每个进程能够打
开的最多文件数。然后使用一个循环,守护程序可以关闭所有可能的文件描
述符。你必须决定你需要做这个或不做。如果你认为有可能有打开的文件描
述符,你需要关闭它们,因为系统有一个同时打开文件数的限制。
7. 为标准输入,标准输出和标准错误输出建立新的文件描述符。即使你不打算
使用它们,打开着它们不失为一个好主意。准确操作这些描述符是基于各自
爱好;比如说,如果你有一个日志文件,你可能希望把它作为标准输出和标
准错误输出打开,而把‘/dev/null’作为标准输入打开;作为替代方法,你可
以将‘/dev/console’作为标准错误输出和/或标准输出打开,而‘/dev/null’作
为标准输入,或者任何其它对你的守护程序有意义的结合方法。(译者注:一
般使用dup2函数原子化关闭和复制文件描述符。
说实话,上面这段文字看着有点云里雾里,下面看个具体的代码(我只粘贴了函数的第一部分,也是最重要的一部分,要查看整个代码,请移步到这
http://www.pythonid.com/bbs/redirect.php?tid=239&goto=lastpost&highlight=自行查看):
def daemonize(stdout='/dev/null', stderr=None, stdin='/dev/null',
pidfile=None, startmsg = 'started with pid %s' ):
'''
This forks the current process into a daemon.
The stdin, stdout, and stderr arguments are file names that
will be opened and be used to replace the standard file descriptors
in sys.stdin, sys.stdout, and sys.stderr.
These arguments are optional and default to /dev/null.
Note that stderr is opened unbuffered, so
if it shares a file with stdout then interleaved output
may not appear in the order that you expect.
'''
# flush io
sys.stdout.flush()
sys.stderr.flush()
# Do first fork.
try:
pid = os.fork()
if pid > 0: sys.exit(0) # Exit first parent.
except OSError, e:
sys.stderr.write("fork #1 failed: (%d) %s\n" % (e.errno, e.strerror))
sys.exit(1)
# Decouple from parent environment.
os.chdir("/")
os.umask(0)
os.setsid()
# Do second fork.
try:
pid = os.fork()
if pid > 0: sys.exit(0) # Exit second parent.
except OSError, e:
sys.stderr.write("fork #2 failed: (%d) %s\n" % (e.errno, e.strerror))
sys.exit(1)
# Open file descriptors and print start message
if not stderr: stderr = stdout
si = file(stdin, 'r')
so = file(stdout, 'a+')
se = file(stderr, 'a+', 0) #unbuffered
pid = str(os.getpid())
sys.stderr.write("\n%s\n" % startmsg % pid)
sys.stderr.flush()
if pidfile: file(pidfile,'w+').write("%s\n" % pid)
# Redirect standard file descriptors.
os.dup2(si.fileno(), sys.stdin.fileno())
os.dup2(so.fileno(), sys.stdout.fileno())
os.dup2(se.fileno(), sys.stderr.fileno())
仔细比对前面说的步骤和代码,大体上是没有什么问题了,但是疑问就来了,os.fork()到底是怎么工作的呢,GOOGLE了个遍,最后的结论是这样:
父进程执行代码到os.fork()处时,会将自己整个拷贝一份(即子进程)这时候父进程os.fork()的返回值大于零(即子进程的PID),子进程os.fork()的返回值等于零,父进程结束,子进程继续执行,这时候又遇到第二个os.fork(),如上次一样,原来的子进程变成了父进程,又产生新的子进程,之后父进程就结束。这就能够说通第一次是避免process group leader,第二次是避免session group leader。子进程就变成了一个五终端,无会话的完全自我掌控的后台进程了。
欢迎大家讨论,不知道是否理解对了。
分享到:
相关推荐
8. **多线程或异步编程**:为了不影响主程序的执行,守护进程的检查和恢复操作可能需要在单独的线程或异步任务中进行。Python的`threading`库或`asyncio`库可以实现这一需求。 通过以上知识点的综合运用,我们可以...
通过redis,获取列表 操作html页面移动
在Python编程中,守护进程(Daemon)是一种特殊类型的后台进程,它不依附于终端,且能在系统启动时启动,并在系统关闭时终止。守护进程常用于提供持续的服务,例如Web服务器、数据库服务等。本篇文章将深入探讨如何...
项目中使用python写一个监控程序,每隔5秒监控目录,发现文件就...为了谨防程序崩溃,特别编写一个守护进程程序,时刻监控程序是否崩溃并重新启动。 博文: http://blog.csdn.net/alex_bean/article/details/77923178
python 守护进程,python使用守护进程运行数据库查询,执行等操作功能
**Python-NutzWk微服务开发框架应用守护进程** 在Python开发领域,微服务架构已经成为构建可扩展、高可用系统的重要手段。NutzWk框架是针对这一需求设计的,它提供了一种高效的微服务开发解决方案。本篇将深入探讨...
守护进程 主进程创建子进程目的是:主进程有一个任务需要并发执行,那开启子进程帮我并发执行任务 主进程创建子进程,然后将该进程设置成守护自己的进程 关于守护进程需要强调两点: 其一:守护进程会在主进程代码...
在Python编程中,守护进程(Daemon)是一种特殊类型的后台进程,通常用于运行长期存在的服务,如Web服务器或数据库服务器。为了有效地管理和控制这些守护进程,开发者通常会编写一个基类来处理守护进程的启动、停止...
同时,守护进程也可以使用其他编程语言来实现,如 Python、Java 等。 Linux 系统中守护进程的创建方法对系统的发展和应用产生了重要的影响。守护进程可以周期性地执行某种任务或等待处理某些特定的事件,使得系统...
守护进程 1、守护子进程 主进程创建守护进程 其一:守护进程会在主进程代码执行结束后就终止 其二:守护进程内无法再开启子进程,否则抛出异常:AssertionError: daemonic processes are not allowed to have...
因为控制台程序出现过因为连接第三方服务超时直接挂掉(此时负载少了一个节点)以及服务器出现意外重启问题(该服务器所有节点宕机),所以需要一个进程守护程序。 也找过 现场的脚本、python的程序、pm2守护方式,...
要使你的python服务不受终端影响而常驻系统,就需要将它变成守护进程。 守护进程就是Daemon程序,是一种在系统后台执行的程序,它独立于控制终端并且执行一些周期任务或触发事件,通常被命名为”d”字母结尾,如...
Python中的守护进程(Daemon)是系统中一种特殊类型的进程,它们不依赖于任何终端,通常在后台运行,用于执行特定的任务,例如日志记录、定时任务等。守护进程在系统启动时启动,并在系统关闭时终止,确保它们在用户...
本文实例讲述了python实现的守护进程(Daemon)用法。分享给大家供大家参考。具体如下: def createDaemon(): "'Funzione che crea un demone per eseguire un determinato programma…"' import os # create - ...
daemonize, 在 python 中,守护进程是一个编写系统守护进程的库 守护进程 守护程序是在 python 中编写系统守护进程的一个库。 它是在MIT许可下发布的。 最新版本可以从 PyPI 下载。 完整的文档可以在 ReadTheDocs ...
实现Python守护进程的步骤涉及多个关键操作,下面将详细解释这些步骤。 1. **创建子进程并退出父进程**: 当创建守护进程时,首先需要通过`os.fork()`创建一个新的子进程。父进程随后退出,使得子进程成为孤儿进程...
python-mpd2, python 库,为音乐播放器守护进程提供客户端接口 python-mpd2 是一个 python 库,它为音乐播放器守护进程提供了一个客户端接口。与 python mpd的区别python-mpd2是 python mpd的fork 。 虽然 0.4. x 向...
### Python守护进程和脚本单例运行详解 #### 一、简介 守护进程是一种常见的后台服务形式,在Linux或Unix环境中非常普遍。它们通常用于提供持续的服务,如日志管理、网络服务等。守护进程需要与启动它的终端环境...
Python multiprocessing.pool创建非守护进程的方法