零基础学python之高级编程(6)---Python中进程的Queue 和进程锁,以及进程池的创建 (包含详细注释代码)

news/2024/4/27 19:19:34

Python中进程的Queue 和进程锁,以及进程池的创建

文章目录

  • Python中进程的Queue 和进程锁,以及进程池的创建
  • 前言
  • 一、进程间同步通信(Queue)
  • 二、进程锁(Lock)
  • 三、创建进程池Poor
      • pool 类方法:
  • End!


前言

大家好,上一篇文章,我们初步接触了进程的概念及其应该如何创建一个进程等等,今天我们继续来深入学习一下进程中的进程间同步 Queue进程锁lock 还有进行创建一个进程池pool 等等,我们开始今天的学习吧 !


一、进程间同步通信(Queue)

我们在使用进程的时候,有时候我们需要进行进程之间的通信,multiprocessing 模块提供了Queue类 来实现不同进程之间的通信.

Queue():括号内传的参数表示最大可接受的消息数量;没有传参或者数量为负数,则表示最大可接受的数量没有上限(直到内存的尽头)

Queue.put(item, block, timeout) : 将item消息写入队列中,block默认为True.

  • 如果block使用默认值,且没有设置timeout秒数,消息队列如果已经没有空间可以写入,将会被阻塞,直到从消息队列腾出空间为止
  • 如果block使用默认值,设置了timeout秒数,则会等待timeout秒,若还没有空间,则会抛出异常’queue.full’.

我们首先先创建一个进程对象:

在这里插入图片描述
上图就是我们queue 中的方法,接下来我们看看如何使用queue.

from  multiprocessing import Queueq = Queue(3)  # 创建一个Queue 对象,设置最大可接受put对象为3个
q.put('消息1')
q.put('消息2')
print(q.full()) # 判断队列是否满了 这里面输出False
q.put('消息3')
print(q.full()) #判断队列是否满了 这里面输出Trueprint(q.get()) # 获取队列中的消息,获取后把消息从队列中删除if not q.full(): # 判断队列是否满,如果不满就继续将消息写入队列q.put('消息4')print(q.get())
print(q.get())
print(q.get())

运行结果:

在这里插入图片描述

from  multiprocessing import Queueq = Queue(3)  # 创建一个Queue 对象,设置最大可接受put对象为3个
q.put('消息1')
q.put('消息2')
print(q.full()) # 判断队列是否满了 这里面输出False
q.put('消息3')
print(q.full()) #判断队列是否满了 这里面输出Trueprint(q.get())
if not q.full():q.put('消息4')# print(q.get())
# print(q.get())
# print(q.get())if not q.empty():  # 判断队列是否为空for i in range(q.qsize()): print(q.get_nowait())

运行结果:
在这里插入图片描述

注意:

Queue.put_nowait(item)===>Queue.put(item, False)

Queue.get_nowait()===>Queue.get(False)

Queue.get(block, timeout):获取队列中的一条信息,然后将其从队列中移除,block默认为True

  • 如果block使用默认值,且没有设置timeout秒数,消息队列如果为空,将会被阻塞,直到从消息队列读到消息为止.

  • 如果block使用默认值,设置了timeout秒数,则会等待timeout秒,若还没有消息,则会抛出异常’queue.empty’.

如果block值为False,消息队列如果为空,则立即抛出异常’queue.empty

接下来我们进行一次完整的进程间的通信

import multiprocessing  #导包
import time
import randomdef write(q):  # 定义一个函数for i in ['a', 'b', 'c']:  print('放入:', i)q.put(i) # 分别将a,b,c 消息传入队列time.sleep(random.random())def read(q):while True:if not q.empty():print('获取:', q.get()) 获取队列中的消息time.sleep(random.random())else:breakif __name__ == '__main__':q = multiprocessing.Queue()  # 创建一个Queue对象 qpw = multiprocessing.Process(target=write, args=(q, ))  # 创建一个子进程 pw ,并将queue 对象传入pr = multiprocessing.Process(target=read, args=(q, ))   # 创建一个子进程 pr,并将queue对象传入pw.start() # pw 子进程开始pw.join()  # 等待 pw进程结束 pr.start() # pr 子进程开始pr.join() # 等待 pr进程结束print('结束')

运行结果:

在这里插入图片描述

二、进程锁(Lock)

Python进程锁是用来在多进程程序中对共享资源进行同步访问的一种机制。
当多个进程需要同时访问某个共享资源时,可能会出现竞争条件(race condition),导致数据不一致或错误的结果。进程锁可以用来保证在任意时刻只有一个进程能够访问共享资源,从而避免竞争条件的发生。

进程锁的主要作用是确保在某个进程正在使用共享资源时,其他进程无法访问该资源,直到当前进程释放锁为止。这样可以保证共享资源在任意时刻只会被一个进程修改,从而避免数据不一致或错误的结果。

Python中的进程锁可以通过使用multiprocessing模块中的Lock类来实现。通过调用lock.acquire()方法可以获得锁,其他进程如果尝试获得同一个锁则会被阻塞。而通过调用lock.release()方法可以释放锁,允许其他进程获得锁并访问共享资源。

需要注意的是,进程锁只能在同一个计算机的多个进程之间起作用,无法在不同计算机之间的进程之间起作用。如果需要进行跨网络的进程同步,可以考虑使用分布式锁的机制。

Python中可以使用多种方式实现进程锁,以下是常用的两种:

使用multiprocessing.Lock():multiprocessing.Lock()是Python标准库中的一个进程锁实现。可以通过acquire()方法获得锁,并在任务完成后使用release()方法释放锁。
例如:

from multiprocessing import Lock, Processdef func(lock, num):lock.acquire()try:# 临界区代码print(f"进程{num}获得了锁")finally:lock.release()print(f"进程{num}释放了锁")if __name__ == "__main__":lock = Lock()processes = []for i in range(5):p = Process(target=func, args=(lock, i))processes.append(p)p.start()for p in processes:p.join()

也可使用multiprocessing.Manager().Lock():multiprocessing.Manager()是一个多进程共享数据的管理器。可以通过Manager().Lock()方法创建一个进程锁。具体使用方式与multiprocessing.Lock()相同,只是创建锁的方式有所不同。
例如:

from multiprocessing import Process, Managerdef func(lock, num):lock.acquire()try:# 临界区代码print(f"进程{num}获得了锁")finally:lock.release()print(f"进程{num}释放了锁")if __name__ == "__main__":manager = Manager()lock = manager.Lock()processes = []for i in range(5):p = Process(target=func, args=(lock, i))processes.append(p)p.start()for p in processes:p.join()

三、创建进程池Poor

Python进程池:是一种用于管理和复用多个进程的工具。进程池可以提高程序的并发处理能力,通过复用已经创建的进程,避免频繁创建和销毁进程的开销。

pool 类方法:

Pool()括号内表示处理的进程数量

apply_async(func, args, kwargs):使用非阻塞方式调用func(并行执行,阻塞就是必须等待上一个进程退出才能执行下一个进程)

close():关闭进程池,不在接收新的进程

terminate():不管任务是否完成,立即终止

join():阻塞,等待所有子进程结束,必须在close之后使用.

import multiprocessing
import timedef run(f):time.sleep(1)return f * fif __name__ == '__main__':test = [1, 2, 3, 4, 5, 6]# print('顺序:')# s = time.time()  # 计算当前时间# for i in test:#     print(run(i))e = time.time()# print('顺序执行时间:', int(e-s))# map(函数名, 可迭代对象):循环将可迭代对象传递给函数执行print('并发:')pool = multiprocessing.Pool(5)  # 创建能够有5条进程的进程池r1 = pool.map(run, test)pool.close()pool.join()e2 = time.time()print('并发时间:', int(e2 - e))print(r1)

运行结果:
在这里插入图片描述

在进程池里进行进程间的通信
例如:

import multiprocessing, os, time, randomdef write(q):print('write(%s)启动' % os.getpid())for i in ['a', 'b', 'c', 'd', 'e']:q.put(i)def read(q):print('read(%s)启动' % os.getpid())for i in range(q.qsize()):print('获取:', q.get())if __name__ == '__main__':print('主进程(%s)开始' % os.getpid())q = multiprocessing.Manager().Queue()pool = multiprocessing.Pool()pool.apply_async(write, (q, ))pool.apply_async(read, (q, ))pool.close()pool.join()print('主进程结束')

运行结果:

在这里插入图片描述


End!

讲的不好,多多见谅,我们下次再见!

更多优质文章点这里


http://www.mrgr.cn/p/65866300

相关文章

C语言从入门到实战----数据在内存中的存储

1. 整数在内存中的存储 在讲解操作符的时候,我们就讲过了下⾯的内容: 整数的2进制表⽰⽅法有三种,即 原码、反码和补码 有符号的整数,三种表⽰⽅法均有符号位和数值位两部分,符号位都是⽤0表⽰“正”,⽤…

NO9 蓝桥杯单片机实践之串口通信的使用

1 回顾 串口通信的代码编写结构还是与中断一样,不同的是: 初始中断函数条件涉及到串口通信相关的寄存器和定时器1相关的寄存器(定时器1用于产生波特率),但初始条件中的中断寄存器只考虑串口通信而不考虑定时器1。 vo…

Matlab|计及电池储能寿命损耗的微电网经济调度

目录 1 主要内容 储能寿命模型 负荷需求响应 2 部分代码 3 程序结果 4 下载链接 1 主要内容 该程序参考文献《考虑寿命损耗的微网电池储能容量优化配置》模型,以购售电成本、燃料成本和储能寿命损耗成本三者之和为目标函数,创新考虑储能寿命损耗约…

程序员如何兼职赚小钱?

程序员由于有技术和手艺其实兼职赚钱的路子还是挺多的,只要你有足够的时间。 1. 做外包 这是比较传统的方式,甲方在一些众包平台上发布开发任务,你可以抢这个任务,但是价格都比较便宜。 任务比较多的平台: 猪八戒、一品威客、开…

C语言指针详解(上)

一.什么是指针 指针是一种类型,用来存储变量的地址的类型 有哪些类型呢 字符指针:char* 整型指针:int* 浮点型指针:float* 双精度浮点型指针:double* 空指针:void* (每一个类型的指针&a…

ubuntu没有有线网络的一种解决办法

直接使用下面的代码: sudo nmcli networking off sudo nmcli networking on转自 Ubuntu 20.04 LTS 有线网络设置消失了怎么办 - 知乎 (zhihu.com) 评论区第一条 理客Lick - 知乎 (zhihu.com)Keep it simple and stupid

Huawei设备基础配置

<Huawei>system-view # 进入系统视图 [Huawei]telnet server enable # 开启设备telnet功能 [Huawei]user-interface vty 0 4 # 开启登录端口0-4并进入用户接口视图 [Huawei-ui-vty0-4]protocol inbound telnet # 通过telnet协议登录 [Huawei-ui-vty0-4]authentic…

如何安装和卸载SFP光模块

SFP光模块的安装和拆卸是简单直接的过程。然而&#xff0c;任何非标准操作都可能导致隐式损坏甚至永久故障。您需要参考及时更新的光模块的数据表或用户手册&#xff0c;以熟悉其特性和锁定机制。 准备工作 常见事项 拆卸和插入SFP光模块可能会缩短其使用寿命&#xff0c;因…

Git工具的详细使用

一、环境说明 [rootgit ~]# getenforce Disabled [rootgit ~]# systemctl status firewalld ● firewalld.service - firewalld - dynamic firewall daemonLoaded: loaded (/usr/lib/systemd/system/firewalld.service; disabled; vendor preset: enabled)Active: inactive (d…

Selenium浏览器自动化测试框架详解

selenium简介 介绍 Selenium [1] 是一个用于Web应用程序测试的工具。Selenium测试直接运行在浏览器中&#xff0c;就像真正的用户在操作一样。支持的浏览器包括IE&#xff08;7, 8, 9, 10, 11&#xff09;&#xff0c;Mozilla Firefox&#xff0c;Safari&#xff0c;Google C…

【转】linux设置密钥登录(只允许密钥登录)

转, 原文: https://www.cnblogs.com/dyj--php/p/9656016.html ----------------- 一.root用户使用这条命令看看家目录有没有(.ssh文件夹),cd .ssh,如果存在就会进入这个文件夹,不存在就(makedir .ssh),修改.ssh这个文件夹权限(chmod 700 .ssh),查看.ssh文件夹下是…

嵌入式作业2.1 汇编练习

目录原始参考代码(main.s)作业一、翻译汇编成C语言二、修改参考代码 原始参考代码(main.s) 汇编代码: //===================================================================== //文件名称:main.s //功能概要:汇编编程调用GPIO构件控制小灯闪烁(利用printf输出提示信…

window10 安装 Redis

一、下载压缩包(免安装) 地址: https://github.com/tporadowski/redis/releases二、解压解压路径自定义 Logs目录,手动添加的,用于保存redis的日志记录 (下面再补充)下图为解压后的文件三、注册Redis服务以管理员权限打开cmd窗口,执行以下命令# 安装redis服务 redis-se…

Camera基础知识四

MIPI-CSI2的PHY层定义了CSI传输介质的电气特性、帧格式以及时钟等,如下:DPHY Lane States:Escape Mode跟csi无关,是DSI用的LP切换到HS模式切换:差分信号一般100mv - 300mv HS_SETTLE,这段时间,1.2V降到100-300mv需要时间稳定下来。HS_ZERO表示发送为0。 一旦MIPITX发送0…

Spring Cloud 十:Spring Cloud与微前端

Spring Cloud 一&#xff1a;Spring Cloud 简介 Spring Cloud 二&#xff1a;核心组件解析 Spring Cloud 三&#xff1a;API网关深入探索与实战应用 Spring Cloud 四&#xff1a;微服务治理与安全 Spring Cloud 五&#xff1a;Spring Cloud与持续集成/持续部署&#xff08;CI/C…

Fiddler(6)AutoResponder

Fiddler最实用的功能, 它可以抓取在线页面保存到本地进行调试, 大大减少了在线调试的困难, 可以让我们修改服务器端返回的数据的

WinRadius 配置指南

WinRadius 配置指南 1 RADIUS概述 RADIUS 是一种用于在需要认证其链接的网络访问服务器(NAS)和共享认证服务器之间进行认证、授权和记帐信息的文档协议。 RADIUS在运维审计系统中,主要体现的是认证功能。 2 设置WinRadius服务器的操作步骤 (1)运行WinRadius。在真实计算机…

Camera基础知识三

参考资料:极客笔记 侵权联删Camera sensor状态机:状态机:POWER OFF、hardware standby、software、streaming 没电的时候就是power off状态,上电了进入hardware standby状态,xshutdown也就是reset,进入software standby状态。PLL寄存器配置进去之后就进入streaming状态Ca…

C语言例4-15:从键盘输入一个整数,求其绝对值并输出。

代码如下&#xff1a; //从键盘输入一个整数&#xff0c;求其绝对值并输出。 #include<stdio.h> int main(void) {int n;printf("输出一个整数&#xff1a; \n");scanf("%d",&n); //从键盘输入一个整数保存至变量nif(n<0) //…