我的编程空间,编程开发者的网络收藏夹
学习永远不晚

Python实现Daemon(守护)进程

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

Python实现Daemon(守护)进程

最近在写Daemon进程,在编写过程中遇到一些小麻烦,最终还是解决了。

我编写了两种,第一种是编写了一个程序,将其用setsid命令让其放入后台运行,第二种是直接fork()一个进程,在代码里将进程设置为后台启动。

在os.sytem()函数其他外部程序时,发现os.system()是阻塞的(os.popen()也是阻塞的),就是启动外部程序,你必须等外部程序退出,它才继续运行。用python中的subprocess库时,发现它并不阻塞主进程的运行,但是,你用外部kill命令杀死进程时,子进程会变成僵尸进程,只有父进程退出后才会退出。网上说在Windows平台下,python有个os.startfile是可以启动外部程序并不阻塞程序的运行,因为我写的Linux环境下,所以该函数不能用。最后问其他朋友,他说可以在system()将输出重定向就可以了,我试了一下,真的可以,所以现在把代码贴出了,也怪自己平常没有怎么钻。

os.system(processName+" 1>/dev/null 2>/dev/null &")

程序功能:

从配置文件读取要监控的进程,对进程实现监控,当监控程序退出时,会自动拉起进程

第一种方法:

在后台启动命令如下:setsid ./WatchProcessDog.py -m &

代码如下:

#!/usr/bin/python
#!encoding=utf-8

import ConfigParser
import sys
import threading
import time
import os
import commands
import subprocess

CONFIG_FILE = "WatchDog.ini"
SECTION="Monitor"
SECTION_KEY="Process"

class CWatchProcess(threading.Thread):
	def __init__(self,configFile):
		threading.Thread.__init__(self)
		self.thread_stop = False
		self.boolexist = False
		self.curPid = os.getpid()
		self.configFile = configFile
		cfg = ConfigParser.ConfigParser()
		try:
			cfg.read(self.configFile)
			allprocesses = cfg.get(SECTION,SECTION_KEY)
			if '#' in allprocesses:
				position1 = allprocesses.find('#')
				self.processes = allprocesses[:position1]
			else:
				self.processes = allprocesses
			self.processes = self.processes.strip()
			self.monitorProcess = self.processes.split(',')
		except Exception,e:
			print e
	def run(self):
		while not self.thread_stop:
			for tmpprocees in self.monitorProcess:
				processname = os.path.basename(tmpprocees)
				count = commands.getoutput("ps -elf | grep %s | grep -v %s | wc -l" % (processname,"grep"))
				if 0 == int(count) and "-m" == sys.argv[1]:
					self.PullProcess(tmpprocees)
					continue
				if 0 == int(count) and "-k" == sys.argv[1]:
					self.boolexist = True
					continue
				if 0 != int(count) and "-m" == sys.argv[1]:
					#print processname+" is started!"
					continue
				if 0 != int(count) and "-k" == sys.argv[1]:
					self.KillProcess(tmpprocees)
					self.boolexist = True
					continue
			if self.boolexist :
				self.KillSelf()
			time.sleep(1)
	def stop(self):
		self.thread_stop = True
	def PullProcess(self,processName):
		os.system(processName+" 1>/dev/null 2>/dev/null &")
	def KillProcess(self,processName):
		os.popen("killall %s" % os.path.basename(processName))
	def KillSelf(self):
		curNum = commands.getoutput("ps -elf | grep %s | grep -v %s | wc -l " % (sys.argv[0],"grep"))
		if 1 == curNum:
			sys.exit()
		elif 1 < curNum:
			result = os.popen("ps -elf | grep %s | grep -v %s" % (sys.argv[0],"grep")).readlines()
			for tmpresult in result:
				tmplist = tmpresult.split()
				if int(self.curPid) == int(tmplist[3]):
					continue
				else:
					os.popen("kill %s" % tmplist[3])
				#tmpcount = len(tmplist)
		sys.exit()
def main():
	count = len(sys.argv)
	if count != 2:
		help()
		sys.exit()
	if "-m" == sys.argv[1]:
		processNum = commands.getoutput("ps -elf | grep %s | grep -v %s | wc -l " % (sys.argv[0],"grep"))
		if 1 < int(processNum):
			print sys.argv[0],"is running" 
			sys.exit()
		print "start monitor processes"
	elif "-k" == sys.argv[1]:
		print "kill all processes"
	else:
		help()
		sys.exit()
def help():
	print "Usage:"
	print "./WatchProcessDog.py -m 			---monitor all processes"
	print "./WatchProcessDog.py -k 			---kill all processes"
if __name__ == "__main__":
	main()
	watchProcess = CWatchProcess(CONFIG_FILE)
	watchProcess.start()
	watchProcess.join()

第二种方法:

启动:./SqyDaemon.py -m

代码如下:

#!/usr/bin/env python
#!encoding=utf-8

import sys, os, time, atexit, string,ConfigParser,commands,subprocess
from signal import SIGTERM 

PID_FILE = "./SqyDaemon.pid"
CONFIG_FILE = "SqyDaemon.ini"
SECTION="Monitor"
SECTION_KEY="Process"

class Daemon:
	def __init__(self,configFile,pidfile):
		self.pidfile = pidfile
		self.configFile = configFile
		cfg = ConfigParser.ConfigParser()
		try:
			cfg.read(self.configFile)
			allprocesses = cfg.get(SECTION,SECTION_KEY)
			if '#' in allprocesses:
				position1 = allprocesses.find('#')
				self.processes = allprocesses[:position1]
			else:
				self.processes = allprocesses
			self.processes = self.processes.strip()
			self.monitorProcess = self.processes.split(',')
		except Exception,e:
			print e
	def _daemonize(self):
		try: 
			pid = os.fork() 
			if pid > 0:
				sys.exit(0)				#退出主进程 
		except OSError, e:
			print "fork failed!\nError is:",e.strerror
			sys.exit(1)
		os.setsid() 
		os.umask(0) 
		#创建子进程
		try: 
			pid = os.fork()
			if pid > 0:
				sys.exit(0) 
		except OSError, e: 
			print "fork failed!\nError is:",e.strerror
			sys.exit(1) 
		#创建processid文件
		atexit.register(self.delpid)
		pid = str(os.getpid())
		file(self.pidfile,'w+').write('%s\n' % pid)
	def delpid(self):
		os.remove(self.pidfile)
	def start(self):
		#检查pid文件是否存在以探测是否存在进程
		try:
			pf = file(self.pidfile,'r')
			pid = int(pf.read().strip())
			pf.close()
		except IOError:
			pid = None
		if pid:
			print "pidfile %s already exist. SqyDaemon already running?\n" % self.pidfile
			sys.exit(1)
		#启动监控
		self._daemonize()
		self._run()
	def stop(self):
		#从pid文件中获取pid
		try:
			pf = file(self.pidfile,'r')
			pid = int(pf.read().strip())
			pf.close()
		except IOError:
			pid = None
		if not pid:
			if "-r" == sys.argv[1]:
				print "SqyDaemon restart and monitor related process!"
			else :
				message = 'pidfile %s does not exist. SqyDaemon not running?\n'
				sys.stderr.write(message % self.pidfile)
			return #重启不报错
		elif "-r" == sys.argv[1]:
			print "%s is runing,now restart!" % sys.argv[0]
		elif "-k" == sys.argv[1]:
			print "all processes are killed!"
		#杀进程
		try:
			while 1:
				os.kill(pid, SIGTERM)
				time.sleep(0.1)
				for tmpprocees in self.monitorProcess:
					processname = os.path.basename(tmpprocees)
					os.system("killall %s" % processname)
		except OSError, err:
			err = str(err)
			if err.find('No such process') > 0:
				if os.path.exists(self.pidfile):
					os.remove(self.pidfile)
			else:
				print str(err)
				sys.exit(1)
	def restart(self):
		self.stop()
		self.start()
	def _run(self):
		while True:
			for tmpprocees in self.monitorProcess:
				processname = os.path.basename(tmpprocees)
				fullpath = os.path.abspath(tmpprocees)
				count = commands.getoutput("ps -elf | grep %s | grep -v %s | wc -l" % (processname,"grep"))
				if 0 == int(count):
					os.system(tmpprocees+" 1>/dev/null 2>/dev/null &") #标准输出和错误输出重定向到/dev/null
				else :
					continue
			time.sleep(2)
def help():
	print "Usage:"
	print "%s -m 			---monitor all processes" % sys.argv[0]
	print "%s -k 			---kill all processes" % sys.argv[0]
	print "%s -r 			---restart all processes" % sys.argv[0]
if __name__ == '__main__':
	daemon = Daemon(CONFIG_FILE,PID_FILE)
	if len(sys.argv) == 2:
		if '-m' == sys.argv[1]:
			daemon.start()
		elif '-k' == sys.argv[1]:
			daemon.stop()
		elif '-r' == sys.argv[1]:
			daemon.restart()
		else:
			print 'Unknown command'
			help()
			sys.exit(2)
		sys.exit(0)
	else:
		help()
		sys.exit(2)


免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

Python实现Daemon(守护)进程

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

Python实现Daemon(守护)进程

最近在写Daemon进程,在编写过程中遇到一些小麻烦,最终还是解决了。我编写了两种,第一种是编写了一个程序,将其用setsid命令让其放入后台运行,第二种是直接fork()一个进程,在代码里将进程设置为后台启动。在os.sytem()函数其
2023-01-31

python daemon守护进程实现

假如写一段服务端程序,如果ctrl+c退出或者关闭终端,那么服务端程序就会退出,于是就想着让这个程序成为守护进程,像httpd一样,一直在后端运行,不会受终端影响。 守护进程英文为daemon,像httpd,mysqld,最后一个字母d其实
2022-06-04

Python守护进程daemon实现

1.1 守护进程守护进程是系统中生存期较长的一种进程,常常在系统引导装入时启动,在系统关闭时终止,没有控制终端,在后台运行。守护进程脱离于终端是为了避免进程在执行过程中的信息在任何终端上显示并且进程也不会被任何终端所产生的终端信息所打断。在
2023-01-31

python 守护进程(daemon)

守护进程的编写步骤:1、fork子进程,然后父进程退出,此时子进程会被init进程接管。2、修改子进程的工作目录,创建新进程组合新会话,修改umask。3、子进程再次fork一个进程,这个进程可以称为孙子进程,然后子进程退出。4、重定向孙子
2023-01-31

python中的daemon守护进程实现

守护进程是生存期长的一种进程。它们独立于控制终端并且周期性的执行某种任务或等待处理某些发生的事件。他们常常在系统引导装入时启动,在系统关闭时终止。守护进程的特性1.在后台运行2.与其运行前的环境隔离开来。这些环境包括未关闭的文件描述符、控制
2023-01-31

Python 守护进程

nohup 可以使程序后台运行不受终端影响,但想使程序运行后就脱离终端Python需要用到os.fork来实现,例子如下:daemonize.py#!/usr/bin/python #coding:utf-8  import sys imp
2023-01-31

python守护进程

假如写一段服务端程序,如果ctrl+c退出或者关闭终端,那么服务端程序就会退出,于是就想着让这个程序成为守护进程,像httpd一样,一直在后端运行,不会受终端影响。守护进程英文为daemon,像httpd,mysqld,最后一个字母d其实就
2023-01-31

[转]Python 守护进程

守护进程:通常被定义为一个后台进程,而且它不属于任何一个终端会话(terminal session)。许多系统服务由守护程序实施;如网络服务,打印等。  下面是转自一位网友写的编写守护进程的步骤: 1. 调用fork()以便父进程可
2023-01-31

Python setdaemon守护进程

setdaemon守护进程#_*_coding:utf-8_*___author__ = 'gaogd'import timeimport threading'''守护进程,如果主线程down了,子线程也就没有了。下面先通过主进程生成mai
2023-01-31

python守护进程监控子进程怎么实现

在Python中,可以使用multiprocessing模块来创建子进程并监控它们。具体实现方法如下:导入multiprocessing模块。import multiprocessing创建一个子进程的函数。def child_proces
2023-10-23

PHP怎么实现守护进程

今天小编给大家分享一下PHP怎么实现守护进程的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。成为守护进程的步骤其实只需要创建子
2023-06-30

C#守护进程如何实现

今天小编给大家分享一下C#守护进程如何实现的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。1、为什么需要守护进程一般是为了保护
2023-07-02

python使用fork实现守护进程的方法

os模块中的fork方法可以创建一个子进程。相当于克隆了父进程 os.fork() 子进程运行时,os.fork方法会返回0;而父进程运行时,os.fork方法会返回子进程的PID号。 所以可以使用PID来区分两个进程:#!/usr/bin
2022-06-04

如何进行Python进程的守护进程实施

如何进行Python进程的守护进程实施,很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。Python进程这一计算机语言在实际的应用中,如果你在实际应用的过程中遇到相
2023-06-17

linux shell实现守护进程脚本

嵌入式初学者,第一次上传代码。昨天做了一个udhcpd与udhcpc的守护,目前只会用shell模仿编写,还有什么方法可以做守护呢?#! /bin/sh #进程名字可修改 PRO_NAME=udhcpc WLAN=ra0while true
2022-06-04

Android通过JNI实现守护进程

开发一个需要常住后台的App其实是一件非常头疼的事情,不仅要应对国内各大厂商的ROM,还需要应对各类的安全管家...虽然不断的研究各式各样的方法,但是效果并不好,比如任务管理器把App干掉,服务就起不来了... 网上搜寻一番后,主要的方法有
2022-06-06

linux 守护进程详解及建立守护进程

linux 守护进程详解及建立守护进程 守护进程是一种后台运行并且独立于所有终端控制之外的进程。守护进程的启动要启动一个守护进程,可以采取一下几种方式:在系统期间通过系统的初始化脚本启动守护进程。这些脚本通常在目录etc/rc.d下,通过它
2022-06-04

Python如何实现守护进程的方法示例

场景设置: 你编写了一个python服务程序,并且在命令行下启动,而你的命令行会话又被终端所控制,python服务成了终端程序的一个子进程。因此如果你关闭了终端,这个命令行程序也会随之关闭。要使你的python服务不受终端影响而常驻系统,就
2022-06-04

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录