"

一、锁机制:  multiprocess.Lock

上篇博客中,我们千方百计实现了程序的异步,让多个任务同时在几个进程中并发处理,但它们之间的运行没有顺序。尽管并发编程让我们能更加充分的利用io资源,但是也给我我们带来了新问题,多个进程使用同一份数据资源的时候,就会引发数据安全或顺序混乱问题. 例:


  1. # 多进程抢占输出资源
  2. from multiprocessing import Process
  3. from os import getpid
  4. from time import sleep
  5. from random import random
  6. def work(i):
  7. print("%s: %s is running" %(i, getpid()))
  8. sleep(random())
  9. print("%s: %s is done" %(i, getpid()))
  10. if __name__ == '__main__':
  11. for i in range(5):
  12. p = Process(target=work, args=(i,))
  13. p.start()

使用互斥锁维护执行顺序:


  1. # 使用锁机制维护执行顺序
  2. from multiprocessing import Process, Lock
  3. from os import getpid
  4. from time import sleep
  5. from random import random
  6. def work(lock, i):
  7. lock.acquire() # ⚠️开锁进门
  8. print("%s: %s is running" %(i, getpid()))
  9. sleep(random())
  10. print("%s: %s is done" %(i, getpid()))
  11. lock.release() # ⚠️出门,归还钥匙
  12. if __name__ == '__main__':
  13. lock = Lock() # ⚠️实例化一把锁,一把钥匙
  14. for i in range(5):
  15. p = Process(target=work, args=(lock, i))
  16. p.start()

上面这种情况虽然使用加锁的形式实现了顺序的执行,却使得进程变成了串行执行,这样确实会浪费些时间,但是保证了数据的安全.

接下来我们以模拟抢票为例,来看看数据安全的重要性:


  1. # 文件db的内容为:{"count": 1} (注意:"count"一定要用双引号,否者json无法识别)
  2. # 并发运行,效率高,但是:竞争抢票只读写同一个文件,造成数据写入错乱
  3. from multiprocessing import Process
  4. from time import sleep
  5. from json import load, dump
  6. def search():
  7. dct = load(open('db'))
  8. print("\033[43m剩余票数:%s\033[0m" %dct['count'])
  9. def get():
  10. dct = load(open('db'))
  11. sleep(0.01) # 模拟读数据的网络延迟
  12. if dct['count'] > 0:
  13. dct['count'] -=1
  14. sleep(0.02) # 模拟写数据的网络延迟
  15. dump(dct, open('db', 'w'))
  16. print("\033[32m购票成功\033[0m")
  17. def tack():
  18. search()
  19. get()
  20. if __name__ == '__main__':
  21. for i in range(100): # 模拟并发100个客户端抢票
  22. p =Process(target=tack)
  23. p.start()
  24. # 100个进程同时读写同一个文件,可能会报错

使用锁机制保护数据安全:


  1. # 文件db的内容为:{"count": 1} (注意:"count"一定要用双引号,否者json无法识别)
  2. # 同步运行,效率低,但是可以保证数据安全
  3. from multiprocessing import Process, Lock
  4. from time import sleep
  5. from json import load, dump
  6. def search():
  7. dct = load(open('db'))
  8. print("\033[43m剩余票数:%s\033[0m" %dct['count'])
  9. def get():
  10. dct = load(open('db'))
  11. sleep(0.01) # 模拟读数据的网络延迟
  12. if dct['count'] > 0:
  13. dct['count'] -=1
  14. sleep(0.02) # 模拟写数据的网络延迟
  15. dump(dct, open('db', 'w'))
  16. print("\033[32m购票成功\033[0m")
  17. def tack():
  18. search()
  19. lock.acquire() # 上锁
  20. get()
  21. lock.release() # 释放
  22. if __name__ == '__main__':
  23. lock = Lock() # 实例化一把锁
  24. for i in range(100): # 模拟并发100个客户端抢票
  25. p =Process(target=tack)
  26. p.start()
  27. # 可能会遇到json.decoder.JSONDecodeError报错

使用锁机制可以保证多个进程想要修改同一块数据时,在同一时间点只能有一个进程进行修改,即串行的修改,牺牲速度而保证安全性.

虽然可以用文件共享数据实现进程间通讯,但问题是:1. 效率低(共享数据基于文件,而文件是硬盘上的数据);2. 需要自己加锁处理。因此我们最好找寻一种解决方案能够兼顾:1. 效率高(多个进程共享一块内存的数据);2. 帮我们处理好锁的问题。这就是mutiprocessing模块为我们提供的基于消息的IPC通信机制:队列和管道

队列和管道都是将数据存放于内存中,而队列又是基于(管道+锁)实现的,可以让我们从复杂的问题中解脱出来,我们应该尽量避免使用共享数据,尽可能使用消息传递和队列,避免处理复杂的同步和锁问题,而且在进程数目增多时,往往可以获得更好的可扩展性.


二、信号量:multiprocessing.Semaphore

1. 信号量Semaphore允许一定数量的线程在同一时间点更改同一块数据,很形象的例子:厕所里有3个坑,同时可以3个人蹲,如果来了第4个人就要在外面等待,直到某个人出来了才能进去.

2.信号量同步基于内部计数器,每调用一次acquire(),计数器-1,每调用一次release(),计数器+1,当计数器为0时,acquire()调用被阻塞。这是迪克斯彻(Dijkstra)信号量概念P()和V()的Python实现,信号同步机制适用于访问像服务器这样的有限资源.

3.注意:信号量与进程池的概念很像,要注意区分,信号量涉及到加锁的概念.

  • 方法

obj = Semaphore(4):实例化一把锁,4把钥匙,钥匙可以指定(最小0,最大未知)

obj.acquire():加锁,锁住下方的语句,直到遇到obj.release()方可解锁

obj.release():释放,释放obj.acquire()和obj.release()的语句数据,此方法如果写在加锁之前,便多了一把钥匙

  • 基本用法

  1. # 一把锁,多把钥匙
  2. from multiprocessing import Semaphore
  3. s = Semaphore(2) # 实例化一把锁,配2把钥匙
  4. s.release() # 可以先释放钥匙,变成3把钥匙
  5. s.release() # 再释放一把钥匙,现在变成了4把钥匙
  6. s.acquire()
  7. print(1)
  8. s.acquire()
  9. print(2)
  10. # 加上先释放的钥匙,我门总共有4把钥匙
  11. s.acquire() # 顺利执行
  12. print(3)
  13. s.acquire() # 顺利执行
  14. print(4)
  15. # 钥匙全部被占用
  16. s.acquire() # 此处阻塞住,等待钥匙的释放
  17. print(5) # 不会被打印
  • 进阶:小黑屋

  1. # 小黑屋
  2. from multiprocessing import Process, Semaphore
  3. from time import sleep
  4. from random import uniform
  5. def func(sem, i):
  6. sem.acquire()
  7. print("第%s个人进入了小黑屋" % i)
  8. sleep(uniform(1, 3))
  9. print("第%s个人走出了小黑屋" % i)
  10. sem.release()
  11. if __name__ == '__main__':
  12. sem = Semaphore(5) # 初始化一把锁,配5把钥匙
  13. for i in range(10): # 启动10个子进程,最多只能5个人同在小黑屋中
  14. p = Process(target=func, args=(sem, i))
  15. p.start()

三、事件机制:multiprocessing.Event

Python线程的事件用于主线程控制其它线程的执行,事件主要提供了三个方法:set、wait,clear

事件处理机制:全局定义了一个“Flag”,如果“Flag”值为False,程序执行wait()方法会被阻塞;如果“Flag”值为True,程序执行wait()方法便不会被阻塞.

  • 方法

obj.is_set():默认值为False,事件是通过此方法的bool值去标示wait()的阻塞状态

obj.set():将is_set()的bool值改为True

obj.clear():将is_set()的bool值改为False

obj.wait():is_set()的值为False时阻塞,否则不阻塞

  • 实例:模拟红绿灯

  1. # 模拟红绿灯
  2. from multiprocessing import Process, Event
  3. import time
  4. import random
  5. def Tra(e):
  6. print("\033[32m绿灯亮\033[0m")
  7. e.set()
  8. while 1:
  9. if e.is_set():
  10. time.sleep(3)
  11. print("\033[31m红灯亮\033[0m")
  12. e.clear()
  13. else:
  14. time.sleep(3)
  15. print("\033[32m绿灯亮\033[0m")
  16. e.set()
  17. def Car(e, i):
  18. e.wait()
  19. print("第%s辆小汽车过去了" % i)
  20. if __name__ == '__main__':
  21. e = Event()
  22. tra = Process(target=Tra, args=(e,))
  23. tra.start()
  24. for i in range(100): # 模拟一百辆小汽车
  25. time.sleep(0.5)
  26. car = Process(target=Car, args=(e, i))
  27. car.start()

"

【Python下进程同步之互斥锁、信号量、事件机制】的更多相关文章

  1. python 并发编程 多进程 互斥锁 目录

    python 并发编程 多进程 互斥锁 模拟抢票 互斥锁与join区别

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

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

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

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

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

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

  5. 《python》join、守护进程、锁/信号量/事件、进程队列

    一.multiprocess.process模块 1.join方法 阻塞主进程,等待子进程执行完毕再放开阻塞 import time import random from multiprocessin ...

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

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

  7. 网络编程并发 多进程 进程池,互斥锁,信号量,IO模型

    进程:程序正在执行的过程,就是一个正在执行的任务,而负责执行任务的就是cpu 操作系统:操作系统就是一个协调.管理和控制计算机硬件资源和软件资源的控制程序. 操作系统的作用: 1:隐藏丑陋复杂的硬件接 ...

  8. python开发进程:互斥锁(同步锁)&进程其他属性&进程间通信(queue)&生产者消费者模型

    一,互斥锁,同步锁 进程之间数据不共享,但是共享同一套文件系统,所以访问同一个文件,或同一个打印终端,是没有问题的, 竞争带来的结果就是错乱,如何控制,就是加锁处理 part1:多个进程共享同一打印终 ...

  9. 谈谈有关 Python 的GIL 和 互斥锁

    转载:https://blog.csdn.net/Amberdreams/article/details/81274217 有 Python 开发经验的人也许听说过这样一句话:Python 不能充分利 ...

随机推荐

  1. linux异常 - 弹出界面 eth0:设备eth0似乎不存在

    问题描述: 用VMware vSphere Client复制虚拟机之后,出现这个问题 解决方法: service network stop service NetworkManager restart

  2. js基础之--变量 作用域和内存问题

    基本类型:Undefind Null Boolean Number String 引用类型: 对象 在操作对象时,实际上实在操作对象的引用而不是实际的对象.为此,引用类型的值是按引用访问的. 从一个变 ...

  3. sql语句代码规范

    19年年底的时候领导一直强调代码规范化以前写代码的时候很随意后来越来越看自己写的代码难受逐渐的也像规范化走去,今天又学了一招记录分享一下 这张图就是以前写代码的时候正常情况很是杂乱无章 这张就是规范话 ...

  4. MySQL判断数据是否为空

    IFNULL(expr1,expr2)函数,这个函数只能判断是否为空 SELECT CONCAT(first_name,',',last_name,',',job_id,IFNULL(commissi ...

  5. 结合字符串常量池/String.intern()/String Table来谈一下你对java中String的理解

    1.字符串常量池 每创建一个字符串常量,JVM会首先检查字符串常量池,如果字符串已经在常量池中存在,那么就返回常量池中的实例引用.如果字符串不在池中,就会实例化一个字符串放到字符串池中.常量池提高了J ...

  6. 命令行(一):Git

    1,使用gitbash进行操作 2,初始化一个Git仓库,使用git init命令. 3,添加文件到Git仓库,分两步:使用命令git add <filename>可反复多次使用,添加多个 ...

  7. 题解【CJOJ1236】【复赛】指数序列求和

    P1236 - [复赛]指数序列求和 Description 求1^b+2^b+…+a^b的和除以10000的余数. Input 第一行包含一个正整数N,表示有N组测试数据接下来N行每行包含两个正整数 ...

  8. 无需QQ成为好友,直接启动QQ客户端聊天

    <a style="color:#fff; margin-left:8px; padding-top:12px;" target="_parent" hr ...

  9. Winform遍历窗口的所有控件(几种方式实现)

    本文链接:https://blog.csdn.net/u014453443/article/details/85088733 扣扣技术交流群:460189483 C#遍历窗体所有控件或某类型所有控件 ...

  10. sql 中联合查询语句

    在查询语句中 两张表进行查询,可以通过 left join (左连接查询) :返回左表中的所有记录和右表中联结字段相等的记录  (意思就是左表中的数据会全部显示,右表中只会显示和左表中相等的字段) r ...