进程同步(multiprocessing.Lock(锁机制)、multiprocessing.Semaphore(信号量机制)、multiprocessing.Event(事件机制))

  在计算机中,有一些硬件和软件,例如处理器、打印机等,都属于竞争类资源,当有需求时,很多进程都要争抢这些资源,而对于这类资源,就属于临界资源。当多进程共同处理某一个数据时,这个数据也就属于一个临界资源。操作系统对计算机内各种资源都使其在竞争中有序化,但是对于数据来说,尤其是用户动态产生的数据,当处理时就变成了临界资源,所以我们作为程序猿来说,需要对临界资源加以保护,否则就会出现数据混乱现象。这是在提高程序效率的优势下,带来的一个隐患。

multiprocessing.Lock(锁机制)

  当多个进程使用同一份数据资源的时候,就会引发数据安全或顺序混乱问题。

from multiprocessing import Process
import random
import time def func(addr):
print('我是%s'%addr)
time.sleep(random.random())
print('谢谢!') if __name__ == '__main__':
l = ['四川的','湖南的','河南的','江苏的']
for addr in l:
p = Process(target=func,args=(addr,))
p.start()
time.sleep(2)
print('\n\n我选%s'%random.choice(l))
# 关于抢占输出资源的事情,是指多进程并发执行时,并不是一个进程执行完任务后其他进程再执行。
# 比如 此程序会输出:我是四川的 我是河南的 我是江苏的 谢谢!谢谢!我是湖南的 谢谢! 谢谢!
# 而不是 : 我是四川的 谢谢! 我是河南的 谢谢! ... 多进程关于抢占输出资源的事情

多进程关于抢占输出资源的事情

from multiprocessing import Process
import random
import time
from multiprocessing import Lock
def func(addr,lock):
lock.acquire()
print('我是%s'%addr)
time.sleep(random.random())
print('谢谢!')
lock.release() if __name__ == '__main__':
lock = Lock()
l = ['四川的','湖南的','河南的','江苏的']
for addr in l:
p = Process(target=func,args=(addr,lock))
p.start()
time.sleep(4)
print('\n\n我选%s'%random.choice(l)) 使用锁维护输出资源

使用锁维护输出资源

上面这种情况,使用了加锁的形式确保了程序的顺序执行,但是执行又变成了串行,降低了效率,但是不得不说,它确保了数据的安全性。

下面举例来说锁的重要性:模拟12306抢票问题。模拟银行账户的存取款问题。

# 注意,文件中存储需要以{'c':1}这种形式,c的引号一定要带
# 否则json识别不出来
# 此代码的效果,并发执行,但是多进程同时读写同一个文件数据,造成数据混乱 from multiprocessing import Process,Lock
import json
import time def check(i,l):
with open('a.txt','r',encoding='utf-8') as f:
dic = json.load(f)
print('第%s个人在查票,余票为%s' % (i, dic['c']))
pay(i,l) def pay(i,l):
with open('a.txt','r',encoding='utf-8') as f:
dic = json.load(f)
time.sleep(0.5)# 模拟网络延迟,当购买过程中也会有网络延迟
if dic['c']:
print('第%s个人买到票了 '%i)
dic['c'] -= 1
else:
print('第%s个人没买到票'%i)
with open('a.txt','w') as f:
json.dump(dic,f) if __name__ == '__main__':
l = Lock()
for i in range(10):
p = Process(target=check,args=(i+1,l))
p.start() 多个人同时抢票

多个人同时抢票

很明显,上述例子中,因为多进程同时对一个临界资源(a.txt文件)进行了读写操作,使文件内数据混乱,也造成了余票为1张,但是很多人都抢到票的假象。那就加锁来解决它吧

from multiprocessing import Process,Lock
import json
import time def check(i,l):
with open('a.txt','r',encoding='utf-8') as f:
dic = json.load(f)
print('第%s个人在查票,余票为%s' % (i, dic['c']))
l.acquire()
pay(i,l)# 为什么在这里加锁? 因为每个人都可以查票,读取数据,不会造成数据混乱,但是当买票的时候,就需要对临界资源的写入,所以对写操作加锁,使某一个进程在写文件时候,其他进程不能碰此文件。
l.release() def pay(i,l):
with open('a.txt','r',encoding='utf-8') as f:
dic = json.load(f)
time.sleep(0.5)# 模拟网络延迟,当购买过程中也会有网络延迟
if dic['c']:
print('第%s个人买到票了 '%i)
dic['c'] -= 1
else:
print('第%s个人没买到票'%i)
with open('a.txt','w') as f:
json.dump(dic,f) if __name__ == '__main__':
l = Lock()
for i in range(10):
p = Process(target=check,args=(i+1,l))
p.start()

加锁解决买票问题

关于银行存取款的问题。同一个账户,某个人一直存,某个人在同一时间一直取,如果不对数据进行保护起来,就会造成的一种数据混乱问题。

from multiprocessing import Process, Lock,Value

def save_money(num):
for i in range(100):
time.sleep(0.05)
num.value += 1 def draw_money(num):
for i in range(100):
time.sleep(0.05)
num.value -= 1 if __name__ == '__main__':
num = Value('i',1000)# 多进程中共享数据,一个int类型的数据,1000
man = Process(target=save_money,args=(num,))
woman = Process(target=draw_money,args=(num,))
man.start()
woman.start()
time.sleep(6)
print(num.value)
from multiprocessing import Process, Lock,Value

def save_money(num,l):
for i in range(100):
time.sleep(0.05)
l.acquire()
num.value += 1
l.release() def draw_money(num,l):
for i in range(100):
time.sleep(0.05)
l.acquire()# 在操作存取款的数据时,先将数据锁住,不允许其他人更改此数据
num.value -= 1
l.release() if __name__ == '__main__':
l = Lock()
num = Value('i',1000)# 多进程中共享数据,一个int类型的数据,1000
man = Process(target=save_money,args=(num,l))
woman = Process(target=draw_money,args=(num,l))
man.start()
woman.start()
time.sleep(6)
print(num.value) 这样才对!!!

multiprocessing.Semaphore(信号量机制)

上述讲的Lock,属于互斥锁,也就是一把钥匙配备一把锁,同时只允许锁住某一个数据。而信号量则是多把钥匙配备多把锁,也就是说同时允许锁住多个数据。

比如在一个粉红发廊,里边有5位服务人员,那么这个发廊最多就同时允许进入5位客人,当又有第6位客人来的时候,就需要在门外等待;当服务人员服务完某位客人后,才允许后续的人再进来一个,换句话说,这个发廊最多同时接待5位客人,多的客人必须等待。

信号量同步基于内部计数器,用户初始化一个计数器初值(比如上述例子中就初始化为5),每调用一次acquire(),计数器减1;每调用一次release(),计数器加1。当计数器为0时,acquire()调用被阻塞。这是迪科斯彻(Dijkstra)信号量概念P()和V()的Python实现。信号量同步机制适用于访问像服务器这样的有限资源。信号量与进程池的概念很像,但是要区分开,信号量涉及到加锁的概念

举个栗子:

from multiprocessing import Semaphore
from multiprocessing import Process
import time
import random def sing(i,se):
se.acquire()# 每次进来一位客人,信号量内部计数器减1
print('%s进入小黑屋'%i)
time.sleep(random.randint(1,3))
print('%s交钱走人'%i)
se.release()# 每次离开一位客人,信号量内部计数器加1 if __name__ == '__main__':
se = Semaphore(5)# 初始化5把钥匙配备5把锁
for i in range(10): # 模拟10个人要进入小黑屋子
p = Process(target=sing,args=(i,se))
p.start()

multiprocessing.Event(事件机制)

python中的事件机制,主要用于主进程控制其他进程的执行,事件主要提供了三个方法 set、wait、clear。

    e = Event()
    e.set() #将is_set()设为True
    e.clear() # 将is_set()设为False
    e.wait() #判断is_set的bool值,如果bool为True,则非阻塞,bool值为False,则阻塞
    e.is_set() # 标识
    事件是通过is_set()的bool值,去标识e.wait() 的阻塞状态
    当is_set()的bool值为False时,e.wait()是阻塞状态
    当is_set()的bool值为True时,e.wait()是非阻塞状态
    当使用set()时,是把is_set的bool变为True
    当使用clear()时,是把is_set的bool变为False

举个栗子:

from multiprocessing import Process, Event
import time def tra(e):
while 1: #红绿灯需要一直亮着,要么红灯,要么绿灯
if e.is_set(): #True代表绿灯了,表示可以过车
time.sleep(5)#睡5秒,让车在这5秒的时间内通过
print('\033[31m红灯亮\033[0m')#绿灯亮5秒后提示红灯亮
e.clear()#把is_set设置为False
else:
time.sleep(5)#此时代表红灯亮,应该红灯亮5秒.在此等5秒
print('\033[32m绿灯亮\033[0m')#红灯亮够5秒该绿灯亮了
e.set()#将is_set设置为True def Car(i,e):
e.wait()#车等在红绿灯,此时要看是红灯还剩绿灯,如果is_set = True 就可以过车
print('第%s辆车过去了' % i) if __name__ == '__main__':
e = Event()
triff_light = Process(target=tra,args=(e,))#信号灯的进程
triff_light.start()
for i in range(50):#描述50辆车的进程
if i % 3 == 0:
time.sleep(2)
car = Process(target=Car,args=(i+1,e,))
car.start()

生产者消费者模型

第一种:

from multiprocessing import Queue,Process

def producer(q,product):
for i in range(20):
info = product + '的娃娃%s号' % i
q.put(info)
q.put(None) def consumer(q,name):
while 1:
info = q.get()
if info:
print('%s拿走了%s' % (name,info))
else:
break if __name__ == '__main__':
q = Queue(20)
p_pro = Process(target=producer,args=(q,'炫彩'))
p_con = Process(target=consumer,args=(q,'corn'))
p_pro.start()
p_con.start()

第二种:

from multiprocessing import Queue,Process

def producer(q,product):
for i in range(20):
info = product + '的娃娃%s号' % str(i)
q.put(info) def consumer(q,name,color):
while 1:
info = q.get()
if info:
print('%s,%s拿走了%s\033[0m' % (color,name,info))
else:# 当消费者获得队列中数据时,如果获得的是None,就是获得到了生产者不再生产数据的标识
break# 此时消费者结束即可 if __name__ == '__main__':
q = Queue()
p_pro1 = Process(target=producer,args=(q,'炫彩'))
p_pro2 = Process(target=producer,args=(q,'苍井井'))
p_pro3 = Process(target=producer,args=(q,'波多多'))
p_con1 = Process(target=consumer,args=(q,'alex','\033[31m'))
p_con2 = Process(target=consumer,args=(q,'wusir','\033[32m'))
p_l = [p_con1,p_con2,p_pro1,p_pro2,p_pro3]
[i.start() for i in p_l]
p_pro1.join()
p_pro2.join()
p_pro3.join()
q.put(None)# 几个消费者就要接受几个结束标识
q.put(None)

进程间通信——队列和管道(multiprocess.Queue、multiprocess.Pipe)

进程间通信--IPC(Inter-Process Communication)

队列(multiprocess.Queue)

import queue  # 不能进行多进程之间的数据传输

(1)from multiprocessing import Queue 借助Queue解决生产者消费者模型,队列是安全的。
  q = Queue(num)
  num : 队列的最大长度
  q.get()# 阻塞等待获取数据,如果有数据直接获取,如果没有数据,阻塞等待
  q.put()# 阻塞,如果可以继续往队列中放数据,就直接放,不能放就阻塞等待

  q.get_nowait()# 不阻塞,如果有数据直接获取,没有数据就报错
  q.put_nowait()# 不阻塞,如果可以继续往队列中放数据,就直接放,不能放就报错

(2)from multiprocessing import JoinableQueue#可连接的队列
  JoinableQueue是继承Queue,所以可以使用Queue中的方法
  并且JoinableQueue又多了两个方法
  q.join()# 用于生产者。等待 q.task_done的返回结果,通过返回结果,生产者就能获得消费者当前消费了多少个数据
  q.task_done() # 用于消费者,是指每消费队列中一个数据,就给join返回一个标识。

管道(multiprocess.Pipe)

from multiprocessing import Pipe

con1,con2 = Pipe()

 管道是不安全的.

 管道是用于多进程之间通信的一种方式.

 如果在单进程中使用管道,那么就是con1收数据,就是con2发数据.

            如果是con1发数据,就是con2收数据

 如果在多进程中使用管道,那么就必须是父进程使用con1收,子进程就必须使用con2发

                  父进程使用con1发,子进程就必须使用con2收

                  父进程使用con2收,子进程就必须使用con1发

                  父进程使用con2发,子进程就必须使用con1收

在管道中有一个著名的错误叫做EOFError.是指,父进程如果关闭了发送端,子进程还继续接收数据,就会产生EOFError错误

进程间的共享内存(Value,Manager)

 from multiprocessing import Manager
 m = Manager()
 num = m.dict({键 : 值})
 num = m.list([1,2,3])

进程池

 含义:

   进程池:一个池子,里边有固定数量的进程。这些进程一直处于待命状态,一旦有任务来,马上就有进程去处理。
 因为在实际业务中,任务量是有多有少的,如果任务量特别的多,不可能要开对应那么多的进程数
 开启那么多进程首先就需要消耗大量的时间让操作系统来为你管理它。其次还需要消耗大量时间让
 cpu帮你调度它。

 进程池还会帮程序员去管理池中的进程。

 方法:

 1).map(func,iterable)

    func:进程池中的进程执行的任务函数
    iterable: 可迭代对象,是把可迭代对象中的每个元素依次传给任务函数当参数

 2).apply(func,args=())同步执行任务

    func:进程池中的进程执行的任务函数
    args: 可迭代对象型的参数,是传给任务函数的参数
    同步处理任务时,不需要close和join
    同步处理任务时,进程池中的所有进程是普通进程(主进程需要等待其执行结束)

 3).apply_async(func,args=(),callback=None)异步执行任务

    func:进程池中的进程执行的任务函数
    args: 可迭代对象型的参数,是传给任务函数的参数
    callback: 回调函数,就是说每当进程池中有进程处理完任务了,返回的结果可以交给回调函数,由回调函数进行进一步的处理,回调函数只有异步才有,同步是没有的
    异步处理任务时,进程池中的所有进程是守护进程(主进程代码执行完毕守护进程就结束)
    异步处理任务时,必须要加上close和join

回调函数

  进程的任务函数的返回值,被当成回调函数的形参接收到,以此进行进一步的处理操作
回调函数是由主进程调用的,而不是子进程,子进程只负责把结果传递给回调函数

python进程-进阶的更多相关文章

  1. python进程进阶

    本节目录: 1.进程的其他方法 2.验证进程之间是空间隔离的 3.守护进程 4.互斥锁 5.编写一个伪抢票程序 6.数据共享 7.for循环,join 8.队列 9.用队列完成一个生产者消费者模型 1 ...

  2. Python之进程 进阶 下

    在python程序中的进程操作 之前我们已经了解了很多进程相关的理论知识,了解进程是什么应该不再困难了,刚刚我们已经了解了,运行中的程序就是一个进程.所有的进程都是通过它的父进程来创建的.因此,运行起 ...

  3. Python语法进阶(1)- 进程与线程编程

    1.进程与多进程 1.1.什么是进程 进程就是程序执行的载体 什么叫多任务? 多任务就是操作系统可以同时运行多个任务.比如你一边在用浏览器学习,还一边在听音乐,,这就是多任务,至少同时有3个任务正在运 ...

  4. Python爬虫进阶五之多线程的用法

    前言 我们之前写的爬虫都是单个线程的?这怎么够?一旦一个地方卡到不动了,那不就永远等待下去了?为此我们可以使用多线程或者多进程来处理. 首先声明一点! 多线程和多进程是不一样的!一个是 thread ...

  5. [ python ] 进程的操作

    目录 (见右侧目录栏导航)- 1. 前言- 2. multiprocess模块- 2.1 multiprocess.Process模块    - 2.2 使用Process模块创建进程    - 2. ...

  6. 年薪20万Python工程师进阶(7):Python资源大全,让你相见恨晚的Python库

    我是 环境管理 管理 Python 版本和环境的工具 pyenv – 简单的 Python 版本管理工具. Vex – 可以在虚拟环境中执行命令. virtualenv – 创建独立 Python 环 ...

  7. python——进程基础

    我们现在都知道python的多线程是个坑了,那么多进程在这个时候就变得很必要了.多进程实现了多CPU的利用,效率简直棒棒哒~~~ 拥有一个多进程程序: #!/usr/bin/env python #- ...

  8. 使用gdb调试Python进程

    使用gdb调试Python进程 有时我们会想调试一个正在运行的Python进程,或者一个Python进程的coredump.例如现在遇到一个mod_wsgi的进程僵死了,不接受请求,想看看究竟是运行到 ...

  9. python进程、线程、协程(转载)

    python 线程与进程简介 进程与线程的历史 我们都知道计算机是由硬件和软件组成的.硬件中的CPU是计算机的核心,它承担计算机的所有任务. 操作系统是运行在硬件之上的软件,是计算机的管理者,它负责资 ...

随机推荐

  1. [SCOI2005]互不侵犯(状压DP)

    嗝~算是状压DP的经典题了~ #\(\mathcal{\color{red}{Description}}\) 在\(N×N\)的棋盘里面放\(K\)个国王,使他们互不攻击,共有多少种摆放方案.国王能攻 ...

  2. Android App的签名打包(晋级篇)

    http://blog.csdn.net/linghu_java/article/details/6701666 Andriod应用程序如果要在手机或模拟器上安装,必须要有签名! 1.签名的意义 为了 ...

  3. DataFrame概念与创建

    一 概念 Pandas是一个开源的Python数据分析库.Pandas把结构化数据分为了三类: Series,1维序列,可视作为没有column名的.只有一个column的DataFrame: Dat ...

  4. HTML5与CSS3网页设计

    <!--一个网页的基本结构写法--> <!doctype html> <html> <head>//头部 <title>标题</tit ...

  5. 身份认证系统(二)多WEB应用的单点登录

    随着互联网的发展,web应用的复杂度也一直在提升,慢慢的单一的web应用已经不能满足复杂的业务需求.例如百度的搜索.新闻.百科.贴吧,其实本质上都是不同的网站.当用户使用这些平台的时候,我们当然不希望 ...

  6. ORACLE逐行累计求和方法(OVER函数)

    1.RANK ( ) OVER ( [QUERY_PARTITION_CLAUSE] ORDER_BY_CLAUSE ) DENSE_RANK ( ) OVER ( [QUERY_PARTITION_ ...

  7. Ionic3环境搭建及创建

    初次尝试Ionic,边学习边记录下来,以免以后忘记了,入坑向( ̄ω ̄;) 1.Ionic环境安装 Ionic开发是依赖于Nodejs环境的,所以在开发之前我们需要安装好Nodejs.下载安装:http ...

  8. 关于JQuery的异步注册

    在采用JQuery进行表单异步提交时,前台传入的是json数据格式,后台controller用map接收,再传回前台进行结果判断时,if-else接收结果()里面,尽量不要出现“=”,不然判断语句失效 ...

  9. c++读取ini的Section节名

    // ConsoleApplication1.cpp : 定义控制台应用程序的入口点.// #include "stdafx.h"#include "iostream&q ...

  10. GCC编译器基础入门

    导语 GCC(GNU Compiler Collection,GNU 编译器套件) 是由 GNU 开发的编程语言编译器,支持C.C++.Objective-C.Fortran.Java.Ada和Go语 ...