Python多线程编程(七):使用Condition实现复杂同步


Posted in Python onApril 05, 2015

目前我们已经会使用Lock去对公共资源进行互斥访问了,也探讨了同一线程可以使用RLock去重入锁,但是尽管如此我们只不过才处理了一些程序中简单的同步现象,我们甚至还不能很合理的去解决使用Lock锁带来的死锁问题。所以我们得学会使用更深层的解决同步问题。

Python提供的Condition对象提供了对复杂线程同步问题的支持。Condition被称为条件变量,除了提供与Lock类似的acquire和release方法外,还提供了wait和notify方法。

使用Condition的主要方式为:线程首先acquire一个条件变量,然后判断一些条件。如果条件不满足则wait;如果条件满足,进行一些处理改变条件后,通过notify方法通知其他线程,其他处于wait状态的线程接到通知后会重新判断条件。不断的重复这一过程,从而解决复杂的同步问题。

下面我们通过很著名的“生产者-消费者”模型来来演示下,在Python中使用Condition实现复杂同步。

''' 

Created on 2012-9-8 

 

@author: walfred 

@module: thread.TreadTest7 

'''  

 

import threading  

import time  

 

condition = threading.Condition()  

products = 0  

 

class Producer(threading.Thread):  

    def __init__(self):  

        threading.Thread.__init__(self)  

 

    def run(self):  

        global condition, products  

        while True:  

            if condition.acquire():  

                if products < 10:  

                    products += 1;  

                    print "Producer(%s):deliver one, now products:%s" %(self.name, products)  

                    condition.notify()  

                else:  

                    print "Producer(%s):already 10, stop deliver, now products:%s" %(self.name, products)  

                    condition.wait();  

                condition.release()  

                time.sleep(2)  

 

class Consumer(threading.Thread):  

    def __init__(self):  

        threading.Thread.__init__(self)  

 

    def run(self):  

        global condition, products  

        while True:  

            if condition.acquire():  

                if products > 1:  

                    products -= 1  

                    print "Consumer(%s):consume one, now products:%s" %(self.name, products)  

                    condition.notify()  

                else:  

                    print "Consumer(%s):only 1, stop consume, products:%s" %(self.name, products)  

                    condition.wait();  

                condition.release()  

                time.sleep(2)  

 

if __name__ == "__main__":  

    for p in range(0, 2):  

        p = Producer()  

        p.start()  

 

    for c in range(0, 10):  

        c = Consumer()  

        c.start()

代码中主要实现了生产者和消费者线程,双方将会围绕products来产生同步问题,首先是2个生成者生产products ,而接下来的10个消费者将会消耗products,代码运行如下:

Producer(Thread-1):deliver one, now products:1

Producer(Thread-2):deliver one, now products:2

Consumer(Thread-3):consume one, now products:1

Consumer(Thread-4):only 1, stop consume, products:1

Consumer(Thread-5):only 1, stop consume, products:1

Consumer(Thread-6):only 1, stop consume, products:1

Consumer(Thread-7):only 1, stop consume, products:1

Consumer(Thread-8):only 1, stop consume, products:1

Consumer(Thread-10):only 1, stop consume, products:1

Consumer(Thread-9):only 1, stop consume, products:1

Consumer(Thread-12):only 1, stop consume, products:1

Consumer(Thread-11):only 1, stop consume, products:1

另外:Condition对象的构造函数可以接受一个Lock/RLock对象作为参数,如果没有指定,则Condition对象会在内部自行创建一个RLock;除了notify方法外,Condition对象还提供了notifyAll方法,可以通知waiting池中的所有线程尝试acquire内部锁。由于上述机制,处于waiting状态的线程只能通过notify方法唤醒,所以notifyAll的作用在于防止有线程永远处于沉默状态。
Python 相关文章推荐
Python中删除文件的程序代码
Mar 13 Python
使用grappelli为django admin后台添加模板
Nov 18 Python
修改Python的pyxmpp2中的主循环使其提高性能
Apr 24 Python
十条建议帮你提高Python编程效率
Feb 16 Python
python-opencv获取二值图像轮廓及中心点坐标的代码
Aug 27 Python
Tensorflow 实现释放内存
Feb 03 Python
Python Tkinter Entry和Text的添加与使用详解
Mar 04 Python
Keras:Unet网络实现多类语义分割方式
Jun 11 Python
python 绘制国旗的示例
Sep 27 Python
使用Python将xmind脑图转成excel用例的实现代码(一)
Oct 12 Python
Python如何实现感知器的逻辑电路
Dec 25 Python
python实现简单石头剪刀布游戏
Oct 24 Python
Python多线程编程(六):可重入锁RLock
Apr 05 #Python
Python多线程编程(五):死锁的形成
Apr 05 #Python
Python多线程编程(四):使用Lock互斥锁
Apr 05 #Python
Python多线程编程(三):threading.Thread类的重要函数和方法
Apr 05 #Python
Python多线程编程(二):启动线程的两种方法
Apr 05 #Python
Python多线程编程(一):threading模块综述
Apr 05 #Python
Python中使用dom模块生成XML文件示例
Apr 05 #Python
You might like
PHP函数之error_reporting(E_ALL ^ E_NOTICE)详细说明
2011/07/01 PHP
php返回相对时间(如:20分钟前,3天前)的方法
2015/04/14 PHP
php  PATH_SEPARATOR判断当前服务器系统类型实例
2016/10/28 PHP
php读取qqwry.dat ip地址定位文件的类实例代码
2016/11/15 PHP
js基于qrcode.js生成二维码的方法【附demo插件源码下载】
2016/12/28 PHP
浅谈Laravel中的三种中间件的作用
2019/10/13 PHP
jquery+json实现的搜索加分页效果
2010/03/31 Javascript
JS实现字体选色板实例代码
2013/11/20 Javascript
关于页面嵌入swf覆盖div层的问题的解决方法
2014/02/11 Javascript
javascript制作的网页侧边弹出框思路及实现代码
2014/05/21 Javascript
JS长整型精度问题实例分析
2015/01/13 Javascript
JS监听微信、支付宝等移动app及浏览器的返回、后退、上一页按钮的事件方法
2016/08/05 Javascript
js中json处理总结之JSON.parse
2016/10/14 Javascript
JS实现一次性弹窗的方法【刷新后不弹出】
2016/12/26 Javascript
js实现上下左右弹框划出效果
2017/03/08 Javascript
JS中mouseup事件丢失的原因与解决办法
2017/06/14 Javascript
使用Vue-Router 2实现路由功能实例详解
2017/11/14 Javascript
vue 实现数字滚动增加效果的实例代码
2018/07/06 Javascript
Vue自定义指令写法与个人理解
2019/02/09 Javascript
JavaScript数据结构与算法之检索算法示例【二分查找法、计算重复次数】
2019/02/22 Javascript
python使用os模块的os.walk遍历文件夹示例
2014/01/27 Python
通过数据库向Django模型添加字段的示例
2015/07/21 Python
python中私有函数调用方法解密
2016/04/29 Python
TensorFlow如何实现反向传播
2018/02/06 Python
python合并同类型excel表格的方法
2018/04/01 Python
对python中的pop函数和append函数详解
2018/05/04 Python
Python3.6.2调用ffmpeg的方法
2019/01/10 Python
Pandas+Matplotlib 箱式图异常值分析示例
2019/12/09 Python
python 轮询执行某函数的2种方式
2020/05/03 Python
Python爬虫之Selenium实现窗口截图
2020/12/04 Python
CHARLES & KEITH台湾官网:新加坡时尚品牌
2019/07/30 全球购物
史学专业毕业生求职信
2014/05/09 职场文书
应届生自荐书
2014/06/23 职场文书
公共场所标语
2014/06/30 职场文书
党员创先争优心得体会
2014/09/11 职场文书
同步小康驻村工作简报
2015/07/20 职场文书