python 34 多进程(二)
1. 互斥锁
当多个进程抢占同一数据时,将数据加锁,使进程按串行的方式去获取数据,先到先得,保证了公平、数据的安全。
lock.acquire() # 加锁
lock.release() # 释放
死锁:连续lock.acquice() 多次,会阻塞进程。
# 模拟三个用户使用同一个打印机打印。
from multiprocessing import Process
from multiprocessing import Lock # 导入互斥锁
import os
import time
import random
import sys
def task1(lock):
lock.acquire() # 加锁
print(f'task1-{os.getpid()}开始打印!')
time.sleep(random.randint(1,3))
print(f'task1-{os.getpid()}打印结束!')
lock.release() # 解锁释放
def task2(lock):
lock.acquire()
print(f'task2-{os.getpid()}开始打印!')
time.sleep(random.randint(1,3))
print(f'task2-{os.getpid()}打印结束!')
lock.release()
def task3(lock):
lock.acquire()
print(f'task3-{os.getpid()}开始打印!')
time.sleep(random.randint(1,3))
print(f'task3-{os.getpid()}打印结束!')
lock.release()
if __name__ == '__main__':
mutex = Lock()
for i in range(1,4):
p = Process(target=getattr(sys.modules[__name__], f'task{i}'), args=(mutex,))
p.start()
# 优化,多个用户打印
from multiprocessing import Process
from multiprocessing import Lock
import os
import time
import random
def task(lock, i):
lock.acquire()
print(f'用户{i}:{os.getpid()}开始打印!')
time.sleep(random.randint(1,3))
print(f'用户{i}:{os.getpid()}打印结束!')
lock.release()
if __name__ == '__main__':
mutex = Lock()
for i in range(1,5):
p = Process(target=task, args=(mutex, i))
p.start()
Lock与join对比:
相同点:都可以把并发变成串行,保证了顺序。
不同点:join是人为设定的顺序;Lock是让其竞争顺序,保证公平性。
2. 进程之间的通信
进程在内存级别是隔离的,但是文件在磁盘上是共享的。
2.1 基于文件的通信
当多个进程共同争抢一个数据、资源时,如果要保证顺序、数据的安全,必须要串行。
缺点:效率低;需人为加锁容易出现死锁。
# 模拟抢票系统,5个用户抢1张票。(查票时是并发的,但购票时是串行的)
# 文件ticket_json 中写入{"count":1}
from multiprocessing import Process
from multiprocessing import Lock
import time
import os
import random
import json
def search(): # 查看余票
time.sleep(random.random())
with open('ticket_json','r', encoding='utf-8') as f1:
dic = json.load(f1)
print(f'{os.getpid()}查看余票:{dic["count"]}')
def paid(): # 购票
with open('ticket_json','r', encoding='utf-8') as f1:
dic = json.load(f1)
if dic["count"] > 0:
dic["count"] -= 1
time.sleep(random.randint(1,2))
with open('ticket_json','w', encoding='utf-8') as f2:
json.dump(dic,f2)
print(f'{os.getpid()}购票成功!')
else:
print(f"{os.getpid()}:已没票!")
def task(lock): # 子进程
search()
lock.acquire() #购票加锁
paid()
lock.release()
if __name__ == '__main__':
mutex = Lock()
for i in range(6): # 5个用户抢1张票
p = Process(target=task, args=(mutex,))
p.start()
2.2 基于队列的通信
队列:存在于内存,可以理解是一个容器。可以承载一些数据。
特性:先进先出,FIFO。
from multiprocessing import Queue
def func():
print('is func')
class Q:
pass
obj = Q()
q = Queue(4) # 最大承载4个数据
q.put(1) #添加数据到队列中
q.put([2])
q.put(func)
q.put(obj)
#q.put(111) # 超出会阻塞
for i in range(5):
print(q.get()) # 依次取出数据,当没数据时,再get会阻塞
# 队列Queue中的一些方法参数
q = Queue(n) # maxsize = n 最大承载n个数据
q.qsize() # 获取队列的元素个数
q.empty() # 判断队列是否为空
q.full() # 判断队列是否满
put(self, obj, block=True, timeout=None)
get(self, block=True, timeout=None)
# 队列满时,再put会阻塞,直到某个进程get()数据时,会添加进去。
# 队列没数据时,再get会阻塞,直到某个进程put()数据时,会取出。
block = True : 默认阻塞。当写block=False时,只有遇到阻塞就会报错。
q.put(11,block=False) # 当队列满时,会报错
q.get(block=False) # 当队列满无数据时时,会报错
timeout = 3 # 阻塞3秒,3秒后还是阻塞状态就会报错。
# 用队列购票
from multiprocessing import Process
from multiprocessing import Queue
import os
import time
import random
def search(q): # 查看余票
print(f"用户-{os.getpid()}查看余票:{q.qsize()}票")
def paid(q): # 购票
if q.qsize() > 0:
q.get()
print(f"用户-{os.getpid()},购票成功")
else:
print(f"用户-{os.getpid()},购票失败")
def task(q):
search(q)
time.sleep(random.random()) # 网络延迟
paid(q)
if __name__ == '__main__':
q = Queue(10)
for i in range(3): # 3张票
q.put(1)
for i in range(5): # 5个用户
p = Process(target=task, args=(q,))
p.start()
python 34 多进程(二)的更多相关文章
- Python编程-多进程二
7.进程间通信(IPC)方式二:管道 (1)创建管道的类: Pipe([duplex]):在进程之间创建一条管道,并返回元组(conn1,conn2),其中conn1,conn2表示管道两端的连接对象 ...
- Python第十二章-多进程和多线程01-多进程
多进程和多线程 一.进程 1.1 进程的引入 现实生活中,有很多的场景中的事情是同时进行的,比如开车的时候 手和脚共同来驾驶汽车,再比如唱歌跳舞也是同时进行的:试想,如果把唱歌和跳舞这2件事情分开依次 ...
- python下实现二叉堆以及堆排序
python下实现二叉堆以及堆排序 堆是一种特殊的树形结构, 堆中的数据存储满足一定的堆序.堆排序是一种选择排序, 其算法复杂度, 时间复杂度相对于其他的排序算法都有很大的优势. 堆分为大头堆和小头堆 ...
- 『Python』多进程处理
尝试学习python的多进程模组,对比多线程,大概的区别在: 1.多进程的处理速度更快 2.多进程的各个子进程之间交换数据很不方便 多进程调用方式 进程基本使用multicore() 进程池优化进程的 ...
- python 使用多进程实现并发编程/使用queue进行进程间数据交换
import time import os import multiprocessing from multiprocessing import Queue, pool ""&qu ...
- Python多线程多进程
一.线程&进程 对于操作系统来说,一个任务就是一个进程(Process),比如打开一个浏览器就是启动一个浏览器进程,打开一个记事本就启动了一个记事本进程,打开两个记事本就启动了两个记事本进程, ...
- Python爬虫实战二之爬取百度贴吧帖子
大家好,上次我们实验了爬取了糗事百科的段子,那么这次我们来尝试一下爬取百度贴吧的帖子.与上一篇不同的是,这次我们需要用到文件的相关操作. 前言 亲爱的们,教程比较旧了,百度贴吧页面可能改版,可能代码不 ...
- python排序之二冒泡排序法
python排序之二冒泡排序法 如果你理解之前的插入排序法那冒泡排序法就很容易理解,冒泡排序是两个两个以向后位移的方式比较大小在互换的过程好了不多了先上代码吧如下: 首先还是一个无序列表lis,老规矩 ...
- 【python】多进程锁multiprocess.Lock
[python]多进程锁multiprocess.Lock 2013-09-13 13:48 11613人阅读 评论(2) 收藏 举报 分类: Python(38) 同步的方法基本与多线程相同. ...
随机推荐
- 月薪12k的零基础自学前端必备手册
随着互联网的深入发展,前端开发工程师一跃成为市场上非常抢手的人才.很多同学,包括以前做UI的.Java的.或者对于IT完全零基础的同学都想学习前端.下图是网上流传甚广的一张前端学习思维导图,很多初学者 ...
- containsObject 总是不含有,你会用吗
结论:containsObject:是在比较内存地址,即使两个对象内容完全一样,地址不同,那也是不同的.我个人认为这个方法应该叫是否存在同一个对象 (开始不知道这个知识,被坑,至少浪费了3个钟头,数组 ...
- web设计_3_可伸缩的导航栏
1. HTML5构建一个选项卡,需要<nav>标签包围一个无序列表,也可以添加role属性告诉辅助设备(如屏幕阅读器)这个元素所扮演的角色. 绝对不要基于图片的导航,对搜索引擎不友好,更新 ...
- 并发栅栏CyclicBarrier---简单问2
并发栅栏CyclicBarrier---简单问 背景:前几天在网上看到关于Java并发包java.concurrent中一个连环炮的面试题,整理下以备不时之需. CyclicBarrier简介: 栅栏 ...
- 【pycharm】pycharm远程连接服务器的Python解释器,远程编写代码!!!
今天讲讲如何用pycharm连接远程服务器,使用远程服务器的Python解释器,比如说是你公司的服务器,在家里就可以编写或修改项目的代码! 第一步,先找到服务器上的ip地址 Linux查看IP命令:i ...
- 什么是https?http升级为https需要什么?
一.什么是https? https是一种加密传输协议,网站使用https后可以避免敏感信息被第三方获取.https加密协议=SSL / TLS+http协议,也就是说,在传统的http协议上加上SSL ...
- solidity智能合约字节数最大值及缩减字节数
智能合约最大字节数 在Solidity中,EIP 170将contract的最大大小限制为24 KB .因此,如果智能合约内容过多,会导致无法进行发布操作. 减少压缩字节数方法 方法及变量命名 在一定 ...
- Python - 自学django,上线一套资产管理系统
一.概述 终于把公司的资产管理网站写完,并通过测试,然后上线.期间包括看视频学习.自己写前后端代码,用时两个多月.现将一些体会记录下来,希望能帮到想学django做web开发的人.大牛可以不用看了,小 ...
- Iphone使用过程中遇到的问题
Q1:同一个Apple ID不同设备之间的通话记录保持同步 解决方法: Step1:"设置"--"电话"--"在其他设备上通话"--选择关闭 ...
- C#实现简单爬虫
分享之前写过的一个爬虫,采集数据,存入数据库的简单实现. github地址:https://github.com/CodesCreator/biu-biu-biu-