一、multiprocess.process模块

1、join方法

  阻塞主进程,等待子进程执行完毕再放开阻塞

import time
import random
from multiprocessing import Process # 单个子进程
def func(index):
time.sleep(random.randint(1, 3))
print('发送完毕') if __name__ == '__main__':
p = Process(target=func, args=(1,))
p.start()
p.join() # 阻塞 直到p进程执行完毕就结束阻塞
print('邮件已经发送完毕')
'''
发送完毕
邮件已经发送完毕
'''

# 多个子进程
def func(index):
time.sleep(random.randint(1,3))
print('第%s个邮件已经发送完毕' % index) if __name__ == '__main__':
p_lst = []
for i in range(10):
p = Process(target=func, args=(i,))
p.start()
p_lst.append(p)
for p in p_lst:
p.join() # 等待每个子进程执行完毕
print('10个邮件已经发送完毕')
'''
第2个邮件已经发送完毕
第4个邮件已经发送完毕
第7个邮件已经发送完毕
第5个邮件已经发送完毕
第1个邮件已经发送完毕
第9个邮件已经发送完毕
第0个邮件已经发送完毕
第3个邮件已经发送完毕
第8个邮件已经发送完毕
第6个邮件已经发送完毕
10个邮件已经发送完毕
'''

2、第二种开启进程的方法

  以继承Process类的形式开启进程的方式

import os
from multiprocessing import Process class MyProcess(Process):
def run(self):
print('子进程: ', os.getpid(), os.getppid()) if __name__ == '__main__':
p = MyProcess()
p.start() # 开启一个子进程,让这个子进程执行run方法
print('主进程:', os.getpid())
'''
主进程: 6852
子进程: 6644 6852
''' # 给子进程传参数 class MyProcess(Process):
def __init__(self, arg):
super().__init__()
self.arg = arg
def run(self):
time.sleep(1)
print('子进程: ', os.getpid(), os.getppid(), self.arg) if __name__ == '__main__':
# 开启单个子进程
p = MyProcess('参数')
p.start() # 开启一个子进程,让这个子进程执行run方法
p.join()
print('主进程:', os.getpid())
'''
子进程: 6552 7784 参数
主进程: 7784
''' if __name__ == '__main__':
# 开启多个子进程
for i in range(10):
p = MyProcess('参数%s' % i)
p.start() # 开启一个子进程,让这个子进程执行run方法
print('主进程:', os.getpid())
'''
主进程: 7340
子进程: 6540 7340 参数0
子进程: 6512 7340 参数3
子进程: 7648 7340 参数1
子进程: 7460 7340 参数2
子进程: 8048 7340 参数4
子进程: 5108 7340 参数8
子进程: 7868 7340 参数7
子进程: 7892 7340 参数6
子进程: 4172 7340 参数9
子进程: 7224 7340 参数5
'''

3、守护进程

  主要功能:每隔一段时间就向一台机器汇报自己的状态(程序的报活)

  特点:会随着主进程的结束而结束。

import time
from multiprocessing import Process def func():
print('子进程 start')
time.sleep(3)
print('子进程 end') if __name__ == '__main__':
p = Process(target=func)
p.daemon = True # 设置p为一个守护进程,必须在start之前完成
p.start()
time.sleep(2)
print('主进程')
'''
子进程 start
主进程
'''
# 主进程会等待子进程完全结束才结束
# 守护进程会随着主进程的代码执行完毕而结束

def func1():
count = 1
while 1:
print(count * '*')
time.sleep(0.5)
count += 1 def func2():
print('func2 start')
time.sleep(5)
print('func2 end') if __name__ == '__main__':
p1 = Process(target=func1)
p1.daemon = True
p1.start() # p1是守护进程
p2 = Process(target=func2)
p2.start()
time.sleep(3)
print('主进程')
'''
func2 start
*
**
***
****
*****
******
主进程
func2 end
'''
# 如果主进程代码已经执行完毕,但是子进程还没执行完,守护进程都不会继续执行
# 守护进程会随着主进程的代码执行完毕而结束
# 主进程会等待子进程结束,守护进程只等待主进程代码结束就结束了

二、进程同步(multiprocess.Lock、multiprocess.Semaphore、multiprocess.Event)

1、锁---multiprocess.Lock

  加锁降低了程序的效率,让原来能够同时执行的代码变成顺序执行了,异步变同步的过程

  好处:保证了数据的安全

import time
import json
from multiprocessing import Process, Lock # 当多个进程使用同一份数据资源的时候,就会引发数据安全或顺序混乱问题。 # 多进程抢占输出资源
def search(person):
with open('ticket') as f:
dic = json.load(f)
time.sleep(0.2) # 模拟网络延迟
print('%s查询余票:' % person, dic['count']) def get_ticket(person):
with open('ticket') as f:
dic = json.load(f)
time.sleep(0.2) # 模拟网络延迟
if dic['count'] > 0:
print('%s买到票了' % person)
dic['count'] -= 1 # 买到票,数量减1
time.sleep(0.2) # 模拟网络延迟
with open('ticket', 'w') as f:
json.dump(dic, f) # 把剩余票数写回文件
else:
print('%s没买到票' % person) def ticket(person):
search(person) # 查票
get_ticket(person) # 抢票 if __name__ == '__main__':
for i in range(10):
p = Process(target=ticket, args=('person%s' % i,))
p.start()
'''
person0查询余票: 5
person4查询余票: 5
person3查询余票: 5
person0买到票了
person8查询余票: 5
person2查询余票: 5
person7查询余票: 5
person4买到票了
person1查询余票: 5
person5查询余票: 5
person3买到票了
person9查询余票: 5
person6查询余票: 5
person8买到票了
person2买到票了
person7买到票了
person1买到票了
person9买到票了
person6买到票了
person5买到票了
''' # 使用锁维护执行顺序
def search(person):
with open('ticket') as f:
dic = json.load(f)
time.sleep(0.2) # 模拟网络延迟
print('%s查询余票:' % person, dic['count']) def get_ticket(person):
with open('ticket') as f:
dic = json.load(f)
time.sleep(0.2) # 模拟网络延迟
if dic['count'] > 0:
print('%s买到票了' % person)
dic['count'] -= 1 # 买到票,数量减1
time.sleep(0.2) # 模拟网络延迟
with open('ticket', 'w') as f:
json.dump(dic, f) # 把剩余票数写回文件
else:
print('%s没买到票' % person) def ticket(person, lock):
search(person)
lock.acquire() # 加锁
get_ticket(person)
lock.release() # 解锁 if __name__ == '__main__':
lock = Lock()
for i in range(10):
p = Process(target=ticket, args=('person%s' % i, lock))
p.start()
'''
person0查询余票: 5
person1查询余票: 5
person2查询余票: 5
person0买到票了
person5查询余票: 5
person6查询余票: 5
person3查询余票: 5
person9查询余票: 5
person4查询余票: 5
person8查询余票: 5
person7查询余票: 5
person1买到票了
person2买到票了
person5买到票了
person6买到票了
person3没买到票
person9没买到票
person4没买到票
person8没买到票
person7没买到票
'''
# 为了保证数据的安全
# 在异步的情况下,多个进程有可能同时修改同一份资源
# 就给这个修改的过程加上锁
import time
from multiprocessing import Process, Lock # 加了锁就把异步变成同步了
def func(num, lock):
time.sleep(1)
print('异步执行', num) # 异步会同时开始 lock.acquire()
time.sleep(0.5)
print('同步执行', num) # 同步要一个结束才开始下一个
lock.release() if __name__ == '__main__':
lock = Lock()
for i in range(10):
p = Process(target=func, args=(i, lock))
p.start()
from multiprocessing import Process, Lock

# 互斥锁
lock = Lock()
lock.acquire()
print('')
lock.acquire()
print('')
# 只打印456,123不打印

2、信号量---multiprocess.Semaphore

  信号量的实现机制:计数器 + 锁 实现的

import time
import random
from multiprocessing import Process, Semaphore def ktv(person, sem):
sem.acquire()
print('\033[32m%s走进ktv\033[0m' % person)
time.sleep(random.randint(1,5))
print('\033[31m%s走出ktv\033[0m' % person)
sem.release() if __name__ == '__main__':
sem = Semaphore(4) # 限定4个
for i in range(10):
p = Process(target=ktv, args=('person%s' % i, sem))
p.start()
'''
person2走进ktv
person1走进ktv
person3走进ktv
person0走进ktv
person1走出ktv
person6走进ktv
person6走出ktv
person7走进ktv
person2走出ktv
person5走进ktv
person3走出ktv
person9走进ktv
person0走出ktv
person8走进ktv
person7走出ktv
person4走进ktv
person8走出ktv
person9走出ktv
person5走出ktv
person4走出ktv
'''

3、事件---multiprocess.Event  

  阻塞事件:wait()方法

    wait 是否阻塞是看 event 对象内部的一个属性

  控制这个属性的值

    set()  将这个属性的值改成True

    clear() 将这个属性的值改成False

    is_set()  判断当前的属性是否为True

import time
import random
from multiprocessing import Process, Event def traffic_light(e):
print('\033[31m红灯亮\033[0m')
while 1:
if e.is_set():
time.sleep(2)
print('\033[31m红灯亮\033[0m')
e.clear()
else:
time.sleep(2)
print('\033[32m绿灯亮\033[0m')
e.set() def car(e, i):
if not e.is_set():
print('car %s 在等待' % i)
e.wait()
print('car %s 通过了' % i) if __name__ == '__main__':
e = Event()
p = Process(target=traffic_light, args=(e,))
p.daemon = True
p.start()
p_lst = []
for i in range(20):
time.sleep(random.randrange(0, 3, 2))
p = Process(target=car, args=(e, i))
p.start()
p_lst.append(p)
for p in p_lst:
p.join()
'''
红灯亮
car 0 在等待
car 1 在等待
绿灯亮
car 1 通过了
car 0 通过了
car 2 通过了
car 3 通过了
car 4 通过了
红灯亮
car 7 在等待
car 6 在等待
car 5 在等待
car 8 在等待
绿灯亮
car 6 通过了
car 7 通过了
car 8 通过了
car 5 通过了
红灯亮
car 9 在等待
绿灯亮
car 9 通过了
car 11 通过了
car 10 通过了
红灯亮
car 12 在等待
car 13 在等待
绿灯亮
car 13 通过了
car 12 通过了
car 15 通过了
car 14 通过了
红灯亮
car 17 在等待
car 16 在等待
car 18 在等待
绿灯亮
car 16 通过了
car 17 通过了
car 18 通过了
car 19 通过了
'''

《python》join、守护进程、锁/信号量/事件、进程队列的更多相关文章

  1. python 全栈开发,Day39(进程同步控制(锁,信号量,事件),进程间通信(队列,生产者消费者模型))

    昨日内容回顾 python中启动子进程并发编程并发 :多段程序看起来是同时运行的ftp 网盘不支持并发socketserver 多进程 并发异步 两个进程 分别做不同的事情 创建新进程join :阻塞 ...

  2. 进程同步控制(锁,信号量,事件), 进程通讯(队列和管道,生产者消费者模型) 数据共享(进程池和mutiprocess.Pool模块)

    参考博客 https://www.cnblogs.com/xiao987334176/p/9025072.html#autoid-1-1-0 进程同步(multiprocess.Lock.Semaph ...

  3. Thread类的其他方法,同步锁,死锁与递归锁,信号量,事件,条件,定时器,队列,Python标准模块--concurrent.futures

    参考博客: https://www.cnblogs.com/xiao987334176/p/9046028.html 线程简述 什么是线程?线程是cpu调度的最小单位进程是资源分配的最小单位 进程和线 ...

  4. python 全栈开发,Day42(Thread类的其他方法,同步锁,死锁与递归锁,信号量,事件,条件,定时器,队列,Python标准模块--concurrent.futures)

    昨日内容回顾 线程什么是线程?线程是cpu调度的最小单位进程是资源分配的最小单位 进程和线程是什么关系? 线程是在进程中的 一个执行单位 多进程 本质上开启的这个进程里就有一个线程 多线程 单纯的在当 ...

  5. python 并发编程 锁 / 信号量 / 事件 / 队列(进程间通信(IPC)) /生产者消费者模式

    (1)锁:进程之间数据不共享,但是共享同一套文件系统,所以访问同一个文件,或同一个打印终端,是没有问题的,而共享带来的是竞争,竞争带来的结果就是错乱,如何控制,就是加锁处理. 虽然使用加锁的形式实现了 ...

  6. python 锁 信号量 事件 队列

    什么是python 进程锁? #同步效率低,但是保证了数据安全  重点 很多时候,我们需要在多个进程中同时写一个文件,如果不加锁机制,就会导致写文件错乱 这个时候,我们可以使用multiprocess ...

  7. python 进程锁 生产者消费者模型 队列 (进程其他方法,守护进程,数据共享,进程隔离验证)

    #######################总结######### 主要理解 锁      生产者消费者模型 解耦用的   队列 共享资源的时候 是不安全的 所以用到后面的锁 守护进程:p.daem ...

  8. python 多线程、线程锁、事件

    1. 多线程的基本使用 import threading import time def run(num): print('Num: %s'% num) time.sleep(3) if num == ...

  9. Python--day38--进程同步控制的---锁\信号量\事件的方法名

随机推荐

  1. 雷林鹏分享:C# 循环

    C# 循环 有的时候,可能需要多次执行同一块代码.一般情况下,语句是顺序执行的:函数中的第一个语句先执行,接着是第二个语句,依此类推. 编程语言提供了允许更为复杂的执行路径的多种控制结构. 循环语句允 ...

  2. 20171104xlVBA各人各科进退

    Sub 各班个人各科进步幅度() Dim dRank As Object Set dRank = CreateObject("Scripting.Dictionary") Dim ...

  3. 动手动脑java异常处理

    1>请阅读并运行AboutException.java示例,然后通过后面的几页PPT了解Java中实现异常处理的基础知识. import javax.swing.*; class AboutEx ...

  4. Spring配置表友好性优化思路

    Spring配置表需要尽量保证对程序员的友好性,一下提供一种优化思路. 中途未保存,心态炸了,只贴图了,fuuuuuuuuuuuuuck 第一种(最烂,最不友好):以Json的格式保存在配置表中,程序 ...

  5. 了解一下express中间件的意思以及next()

    app.use()就是通常所说的使用中间件 一个请求发送到服务器后,它的生命周期是 先收到request(请求),然后服务端处理,处理完了以后发送response(响应)回去,而这个服务端处理的过程就 ...

  6. python中RabbitMQ的使用(路由键)

    1.简介 当我们希望每个接收端接收各自希望的消息时,我们可以使用路由键,此时交换机的类型为direct. 2.工作原理 每个接收端的消息队列在绑定交换机的时候,可以设定相应的路由键. 发送端通过交换机 ...

  7. zzw原创_Red Hat Enterprise Linux Server release 6.5 安装mysql5.5.28版本

    1.查看系统版本 [root@ip12189 ~]#  more /etc/issueRed Hat Enterprise Linux Server release 6.5 (Santiago)Ker ...

  8. ORA-12537:TNS:connectionclosed错误处理过程

    1.ORA-12537:TNS:connectionclosed错误处理过程 检查监听正常,oracle服务也是正常启动的,但是登录不进去. 2.解决方案 1. cd $ORACLE_HOME/bin ...

  9. Django之自定义分页

    分页功能在每个网站都是必要的,对于分页来说,其实就是根据用户的输入计算出应该显示在页面上的数据在数据库表中的起始位置. 1. 每页显示的数据条数 2. 每页显示页号链接数 3. 上一页和下一页 4. ...

  10. vue 小知识

    图片: 1.img 的路径 <img :src="item.src"/> 2.背景图片的路径 v-bind:style="{backgroundImage: ...