运行环境:Ubuntu16.04 (fork在Windows系统里不能使用)
孤儿和僵尸进程
孤儿进程 : 父进程先于子进程退出,此时子进程成为孤儿进程。
特点: 孤儿进程会被系统进程收养,此时系统进程就会成为孤儿进程新的父进程,孤儿进程退出该进程会自动处理。
僵尸进程 : 子进程先于父进程退出,父进程又没有处理子进程的退出状态,此时子进程就会称为僵尸进程。
特点: 僵尸进程虽然结束,但是会存留部分PCB在内存中,大量的僵尸进程会浪费系统的内存资源。
如何避免僵尸进程产生
1.使用wait函数处理子进程退出
pid,status = os.wait()功能:在父进程中阻塞等待处理子进程退出返回值: pid 退出的子进程的PIDstatus 子进程退出状态 等于退出状态*256
2.创建二级子进程处理僵尸
代码示例:child.py
【1】 父进程创建子进程,等待回收子进程
【2】 子进程创建二级子进程然后退出
【3】 二级子进程称为孤儿,和原来父进程一同执行事件
"""
创建二级子进程处理僵尸
"""
import os
from time import sleep
def f1():
for i in range(3):
sleep(2)
print("写代码")
def f2():
for i in range(2):
sleep(4)
print("测代码")
# 将二级子进程变成孤儿,自动由系统处理
pid = os.fork()
if pid == 0:
p = os.fork() # 创建二级子进程
if p == 0:
f1()
else:
os._exit(0) # 一级子进程退出
else:
os.wait() # 等待回收一级子进程
f2()
3.通过信号处理子进程退出
原理: 子进程退出时会发送信号给父进程,如果父进程忽略子进程信号,则系统就会自动处理子进程退出。
方法: 使用signal模块在父进程创建子进程前写如下语句 :
import signal
signal.signal(signal.SIGCHLD,signal.SIG_IGN)
特点 : 非阻塞,不会影响父进程运行。可以处理所有子进程退出
"""
模拟僵尸进程产生
"""
import os,sys
import signal
# 忽略子进程的退出行为,子进程退出自动由系统处理
signal.signal(signal.SIGCHLD,signal.SIG_IGN)
pid = os.fork()
if pid < 0:
print("Error")
elif pid == 0:
print("Child PID:",os.getpid())
sys.exit(2)
else:
"""
os.wait() 处理僵尸
"""
# pid,status = os.wait()
# print("pid:",pid)
# print('status:',os.WEXITSTATUS(status))
while True: # 让父进程不退出
pass