Python高级编程之消息队列(Queue)与进程池(Pool)实例详解
作者:HMMHMH 发布时间:2021-12-14 17:30:24
标签:Python,消息队列,Queue,进程池,Pool
本文实例讲述了Python高级编程之消息队列(Queue)与进程池(Pool)。分享给大家供大家参考,具体如下:
Queue消息队列
1.创建
import multiprocessing
queue = multiprocessing.Queue(队列长度)
2.方法
方法 | 描述 |
---|---|
put | 变量名.put(数据),放入数据(如队列已满,则程序进入阻塞状态,等待队列取出后再放入) |
put_nowait | 变量名.put_nowati(数据),放入数据(如队列已满,则不等待队列信息取出后再放入,直接报错) |
get | 变量名.get(数据),取出数据(如队列为空,则程序进入阻塞状态,等待队列防如数据后再取出) |
get_nowait | 变量名.get_nowait(数据),取出数据(如队列为空,则不等待队列放入信息后取出数据,直接报错),放入数据后立马判断是否为空有时为True,原因是放入值和判断同时进行 |
qsize | 变量名.qsize(),消息数量 |
empty | 变量名.empty()(返回值为True或False),判断是否为空 |
full | 变量名.full()(返回值为True或False),判断是否为满 |
3.进程通信
因为进程间不共享全局变量,所以使用Queue进行数据通信,可以在父进程中创建两个字进程,一个往Queue里写数据,一个从Queue里取出数据。
例:
import multiprocessing
import time
def write_queue(queue):
# 循环写入数据
for i in range(10):
if queue.full():
print("队列已满!")
break
# 向队列中放入消息
queue.put(i)
print(i)
time.sleep(0.5)
def read_queue(queue):
# 循环读取队列消息
while True:
# 队列为空,停止读取
if queue.empty():
print("---队列已空---")
break
# 读取消息并输出
result = queue.get()
print(result)
if __name__ == '__main__':
# 创建消息队列
queue = multiprocessing.Queue(3)
# 创建子进程
p1 = multiprocessing.Process(target=write_queue, args=(queue,))
p1.start()
# 等待p1写数据进程执行结束后,再往下执行
p1.join()
p1 = multiprocessing.Process(target=read_queue, args=(queue,))
p1.start()
执行结果:
Pool进程池
初始化Pool时,可以指定一个最大进程数,当有新的请求提交到Pool中时,如果池还没有满,那么就会创建一个新的进程用来执行该请求;但如果池中的进程数已经达到指定的最大值,那么该请求就会等待,直到池中有进程结束,才会用之前的进程来执行新的任务。
1.创建
import multiprocessing
pool = multiprocessing.Pool(最大进程数)
2.方法
方法 | 描述 |
---|---|
apply() | 以同步方式添加进程 |
apply_async() | 以异步方式添加进程 |
close() | 关闭Pool,使其不接受新任务(还可以使用) |
terminate() | 不管任务是否完成,立即终止 |
join() | 主进程阻塞,等待子进程的退出,必须在close和terminate后使用 |
3.进程池内通信
创建进程池内Queue消息队列通信
import multiprocessing
Queue:queue = multiprocessing.Manager().Queue()
例:
import multiprocessing
import time
写入数据的方法
def write_data(queue):
# for循环 向消息队列中写入值
for i in range(5):
# 添加消息
queue.put(i)
print(i)
time.sleep(0.2)
print("队列已满~")
创建读取数据的方法
def read_data(queue):
# 循环读取数据
while True:
# 判断队列是否为空
if queue.qsize() == 0:
print("队列为空~")
break
# 从队列中读取数据
result = queue.get()
print(result)
if __name__ == '__main__':
# 创建进程池
pool = multiprocessing.Pool(2)
# 创建进程池队列
queue = multiprocessing.Manager().Queue()
# 在进程池中的进程间进行通信
# 使用线程池同步的方式,先写后读
# pool.apply(write_data, (queue, ))
# pool.apply(read_data, (queue, ))
# apply_async() 返回ApplyResult 对象
result = pool.apply_async(write_data, (queue, ))
# ApplyResult对象的wait() 方法,表示后续进程必须等待当前进程执行完再继续
result.wait()
pool.apply_async(read_data, (queue, ))
pool.close()
# 异步后,主线程不再等待子进程执行结束,再结束
# join() 后,表示主线程会等待子进程执行结束后,再结束
pool.join()
运行结果:
4.案例(文件夹copy器)
代码:
# 导入模块
import os
import multiprocessing
# 拷贝文件函数
def copy_dir(file_name, source_dir, desk_dir):
# 要拷贝的文件路径
source_path = source_dir+'/'+file_name
# 目标路径
desk_path = desk_dir+'/'+file_name
# 获取文件大小
file_size = os.path.getsize(source_path)
# 记录拷贝次数
i = 0
# 以二进制度读方式打开原文件
with open(source_path, "rb") as source_file:
# 以二进制写入方式创建并打开目标文件
with open(desk_path, "wb") as desk_file:
# 循环写入
while True:
# 读取1024字节
file_data = source_file.read(1024)
# 如果读到的不为空,则将读到的写入目标文件
if file_data:
desk_file.write(file_data)
# 读取次数+1
i += 1
# 拷贝百分比进度等于拷贝次数*1024*100/文件大小
n = i*102400/file_size
if n >= 100:
n = 100
print(file_name, "拷贝进度%.2f%%" % n)
else:
print(file_name, "拷贝成功")
break
if __name__ == '__main__':
# 要拷贝的文件夹
source_dir = 'test'
# 要拷贝到的路径
desk_dir = 'C:/Users/Administrator/Desktop/'+source_dir
# 存在文件夹则不创建
try:
os.mkdir(desk_dir)
except:
print("目标文件夹已存在,未创建")
# 获取文件夹内文件目录,存到列表里
file_list = os.listdir(source_dir)
print(file_list)
# 创建进程池,最多同时运行3个子进程
pool = multiprocessing.Pool(3)
for file_name in file_list:
# 异步方式添加到进程池内
pool.apply_async(copy_dir, args=(file_name, source_dir, desk_dir))
# 关闭进程池(停止添加,已添加的还可运行)
pool.close()
# 让主进程阻塞,等待子进程结束
pool.join()
运行结果:
希望本文所述对大家Python程序设计有所帮助。
来源:https://blog.csdn.net/zsh142537/article/details/82556147


猜你喜欢
- 有如下的一堆mac地址,需要更改成一定格式,如mac='902B345FB021'改为mac='90-2B-34-5
- Python2.7对于中文编码的问题处理的并不好,这几天在爬数据的时候经常会遇到中文的编码问题。但是本人对编码原理不了解,也没时间深究其中的
- CSS3的box-shadow属性可以让我们轻松实现图层阴影效果。我们来实战详解一下这个属性。1. box-shadow属性的浏览器兼容性先
- python中使用requests模块http请求时,发现中文参数不会自动的URL编码,并且没有找到类似urllib (python3)模块
- 本文实例为大家分享了HTML5 JS压缩图片,并获取图片BASE64编码上传的方法,供大家参考,具体内容如下基本过程1) 调用 FileRe
- js原生获取css样式,并且设置,看似简单,其实并不简单,我们平时用的ele.style.样式,只能获取内嵌的样式,但是我们写的样式基本都在
- 今天用python实现了一下简单的聚类分析,顺便熟悉了numpy数组操作和绘图的一些技巧,在这里做个记录。from pylab import
- 基于python的Appium进行b站直播消费记录爬取之前看文章说fiddler也可以进行爬取,但尝试了一下没成功,这次选择appium进行
- 中间那个控制块,其实也是一个iframe,把他的宽度定义为10。 然后在他的内部js,控制 左右2个iframe. functio
- 在JS中有些内存只需执行一遍即可,如浏览器类型检测是最常用的一个功能,因为我们使用Ajax的时候需要检测浏览器的内置的XHR。我们可以在第一
- 一.安装环境:Linux系统: CentOS 6.3 64位Oracle: Oracl
- window对象是JavaScript浏览器对象模型中的顶层对象,包含多个常用方法和属性: 1 打开新窗口 window.open(page
- 前言:前面提到了Python中的数值型内置数据类型,接下来呢我们就着重介绍一下字符串类型。在Python中字符串是一个有序的字符集合,没有独
- 处理过滤Apache日志文件access_test.log文件内容27.19.74.143 - - [30/May/2015:17:38:2
- 前提1.python环境及tensorflow安装成功2.Anaconda安装好 ,Anaconda安装步骤安装步骤1.下载facenet,
- 1. OS标准库简介顾名思义,OS表示Operating System,即操作系统。OS标准库是一个操作系统接口模块,提供一些方便使用操作系
- 二进制日志的文件的作用 mysql二进制日志文件用来记录所有用户对数据库操作,即记录用户对数据库操作的
- 管理界面是基础设施中非常重要的一部分。这是以网页和有限的可信任管理者为基础的界面,它可以让你添加,编辑和删除网站内容。Django有自己的自
- 共同点两者都接收两个参数,第一个参数是行的范围,第二个参数是列的范围不同点loc函数接收的是行/列的名称,iloc函数接收的是行/列的下标(
- Go语言作为一门开源的编程语言,已经广泛应用于各个领域。作为一门现代化的编程语言,Go语言支持模块化开发,而包和依赖管理是模块化开发的重要组