qq 守护进程 android,守护进程

守护进程

1、定义:

守护进程:脱离终端并且在后台运行的进程

守护进程脱离终端:避免进程在执行过程中的信息在任何终端上显示;进程不会被任何终端所产生的终端信息所打断

2、创建步骤:

创建子进程,退出父进程:

使用fork()函数和if判断语句,使子进程变为孤儿进程,交给init进程管理

pid = os.fork()         #调用fork()函数

if pid 

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 

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

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值