守护进程
2015-07-03 10:42
267 查看
守护进程1、定义: 守护进程:脱离终端并且在后台运行的进程 守护进程脱离终端:避免进程在执行过程中的信息在任何终端上显示;进程不会被任何终端所产生的终端信息所打断
2、创建步骤: 创建子进程,退出父进程: 使用fork()函数和if判断语句,使子进程变为孤儿进程,交给init进程管理
3、一个创建守护进程的实例:
4、停止守护进程的方法: 使用kill杀死
python守护进程的完整实例:
另外说明:
umask()函数为进程设置文件模式创建屏蔽字,并返回以前的值
在shell命令行输入:umask 就可知当前文件模式创建屏蔽字
常见的几种umask值是002,022和027,002阻止其他用户写你的文件,022阻止同组成员和其他用户写你的文件,027阻止同组成员写你的文件以及其他用户读写或执行你的文件
rwx-rwx-rwx 代表是777 所有的人都具有权限读写与执行
chmod()改变文件的权限位
int dup(int filedes) 返回新文件描述符一定是当前文件描述符中的最小数值
int dup2(int filedes, int filedes2);这两个函数返回的新文件描述符与参数filedes共享同一个文件表项。
sys.stdout.flush():python的输出(stdout,stderr)是有缓冲区的
flush()方法会直接把内部缓冲区的数据立刻写入文件,而不是被动的等待输出缓冲区被写入
obj.fileno():获取打开文件的描述符
os.dup2(f1,f2):复制f1的文件描述符到f2
2、创建步骤: 创建子进程,退出父进程: 使用fork()函数和if判断语句,使子进程变为孤儿进程,交给init进程管理
pid = os.fork() #调用fork()函数 if pid < 0: #小于0,表明调用fork失败 print 'invoke fork() failure' sys.exit(1) #退出脚本 elif pid > 0: #大于0,表明是父进程 sys.exit(0) #退出父进程在子进程中创建新会话: 这个步骤是创建守护进程中最重要的一步;调用fork函数创建子进程时,子进程继承了父进程的全部资源环境(包括会话期、进程组、控制终端等),虽然父进程退出了,但会话期、进程组、控制终端等并没有改变,因此,子进程并没有真正的独立出来,而setsid函数能够使子进程完全独立出来
os.setsid() #调用setsid()函数改变当前目录为根目录: 调用fork()函数创建子进程,子进程也会继承父进程的当前工作目录,若子进程使用父进程的当前工作目录可能会有一些问题
os.chdir("/")重设文件权限掩码: 把文件权限掩码设置为0,可以大大增强守护进程的灵活性
os.umask(0)
3、一个创建守护进程的实例:
vi /tmp/dir1/file.py #encoding:utf-8 import os import sys import time import commands pid = os.fork() #调用fork()函数 if pid < 0: #小于0,表面调用fork失败 print 'invoke fork() failure' sys.exit(1) #退出脚本 elif pid > 0: #大于0,表面是父进程 sys.exit(0) #退出父进程,下面所有命令都是在子进程下执行 os.setsid() #在子进程中调用setsid函数 os.chdir("/") os.umask(0) while True: os.system('echo `date +%F-%H%M%S` >> /tmp/dir1/file') time.sleep(1)
[root@scj dir1]# python /tmp/dir1/file.py [root@scj dir1]# ps -ef | grep file.py root 3282 1 0 02:46 ? 00:00:00 python /tmp/dir1/file.py root 3311 1575 0 02:46 pts/1 00:00:00 grep file.py由上发现:file.py被放在后台作为守护进程运行,父进程号是1(也就是init进程)注意:守护进程由init进程管理
[root@scj dir1]# tail -f /tmp/dir1/file 2015-06-23-025041 2015-06-23-025042 2015-06-23-025043 2015-06-23-025044 2015-06-23-025045 2015-06-23-025046 2015-06-23-025047 2015-06-23-025048 2015-06-23-025049 2015-06-23-025050 2015-06-23-025051 2015-06-23-025052 2015-06-23-025053 2015-06-23-025054每秒执行一次
4、停止守护进程的方法: 使用kill杀死
[root@scj dir1]# ps -ef | grep file.py root 3282 1 0 02:46 ? 00:00:00 python /tmp/dir1/file.py root 4195 1575 0 02:51 pts/1 00:00:00 grep file.py [root@scj dir1]# kill -9 3282
python守护进程的完整实例:
# -*-coding:utf-8-*- import sys, os '''将当前进程fork为一个守护进程 注意:如果你的守护进程是由inetd启动的,不要这样做!inetd完成了 所有需要做的事情,包括重定向标准文件描述符,需要做的事情只有 chdir() 和 umask()了 ''' def daemonize(stdin='/dev/null',stdout= '/dev/null', stderr= 'dev/null'): '''Fork当前进程为守护进程,重定向标准文件描述符 (默认情况下定向到/dev/null) ''' #Perform first fork. try: pid = os.fork() if pid > 0: sys.exit(0) #first parent out except OSError, e: sys.stderr.write("fork #1 failed: (%d) %s\n" %(e.errno, e.strerror)) sys.exit(1) #从母体环境脱离 os.chdir("/") os.umask(0) os.setsid() #执行第二次fork try: pid = os.fork() if pid > 0: sys.exit(0) #second parent out except OSError, e: sys.stderr.write("fork #2 failed: (%d) %s]n" %(e.errno,e.strerror)) sys.exit(1) #进程已经是守护进程了,重定向标准文件描述符 for f in sys.stdout, sys.stderr: f.flush() si = file(stdin, 'r') so = file(stdout,'a+') se = file(stderr,'a+',0) os.dup2(si.fileno(), sys.stdin.fileno()) os.dup2(so.fileno(), sys.stdout.fileno()) os.dup2(se.fileno(), sys.stderr.fileno()) def _example_main(): '''示例函数:每秒打印一个数字和时间戳''' import time sys.stdout.write('Daemon started with pid %d\n' % os.getpid()) sys.stdout.write('Daemon stdout output\n') sys.stderr.write('Daemon stderr output\n') c = 0 while True: sys.stdout.write('%d: %s\n' %(c, time.ctime())) sys.stdout.flush() c = c+1 time.sleep(1) if __name__ == "__main__": daemonize('/dev/null','/home/hzhida/daemon.log','home/hzhida/daemon.log') _example_main()#第一个fork是为了让shell返回,同时让你完成setsid(从你的控制终端移除,这样就不会意外地收到信号)。setsid使得这个进程成为“会话领导(session leader)”,即如果这个进程打开任何终端,该终端就会成为此进程的控制终端。我们不需要一个守护进程有任何控制终端,所以我们又fork一次。在第二次fork之后,此进程不再是一个“会话领导”,这样它就能打开任何文件(包括终端)且不会意外地再次获得一个控制终端
另外说明:
umask()函数为进程设置文件模式创建屏蔽字,并返回以前的值
在shell命令行输入:umask 就可知当前文件模式创建屏蔽字
常见的几种umask值是002,022和027,002阻止其他用户写你的文件,022阻止同组成员和其他用户写你的文件,027阻止同组成员写你的文件以及其他用户读写或执行你的文件
rwx-rwx-rwx 代表是777 所有的人都具有权限读写与执行
chmod()改变文件的权限位
int dup(int filedes) 返回新文件描述符一定是当前文件描述符中的最小数值
int dup2(int filedes, int filedes2);这两个函数返回的新文件描述符与参数filedes共享同一个文件表项。
sys.stdout.flush():python的输出(stdout,stderr)是有缓冲区的
flush()方法会直接把内部缓冲区的数据立刻写入文件,而不是被动的等待输出缓冲区被写入
obj.fileno():获取打开文件的描述符
os.dup2(f1,f2):复制f1的文件描述符到f2
相关文章推荐
- php守护进程 加linux命令nohup实现任务每秒执行一次
- Python守护进程(daemon)代码实例
- C语言编写Linux守护进程实例
- linux shell实现守护进程脚本
- linux下的守护进程
- PHP守护进程实例
- PHP程序级守护进程的实现与优化的使用概述
- PHP扩展程序实现守护进程
- shell脚本作为保证PHP脚本不挂掉的守护进程实例分享
- PHP高级编程实例:编写守护进程
- 使用Python编写Linux系统守护进程实例
- linux 创建守护进程的相关知识
- Python守护进程用法实例分析
- 进程管理工具supervisor简介及使用技巧
- PHP守护进程实例
- Linux操作系统下守护进程的编程方法
- 守护进程
- Linux守护进程简介
- 守护进程和inetd超级服务器
- Linux 守护进程创建