Python多线程 threading模块[8] Condition实现复杂同步

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

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

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

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

# !/usr/bin/python3
# coding: utf-8
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("生产者 %s 生产,剩余商品: %s个" % (self.name, products))
                    condition.notify()
                else:
                    print("生产者 %s 等待,剩余商品: %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 > 0:
                    products -= 1
                    print("消费者 %s 消费,剩余商品: %s个" % (self.name, products))
                    condition.notify()
                else:
                    print("消费者 %s 等待,剩余商品: %s个" % (self.name, products))
                    condition.wait()
                condition.release()
                time.sleep(2)


if __name__ == "__main__":
    for p in range(0, 3):
        p = Producer()
        p.start()

    for c in range(0, 6):
        c = Consumer()
        c.start()

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

生产者 Thread-1 生产,剩余商品: 1个
生产者 Thread-2 生产,剩余商品: 2个
生产者 Thread-3 生产,剩余商品: 3个
消费者 Thread-4 消费,剩余商品: 2个
消费者 Thread-5 消费,剩余商品: 1个
消费者 Thread-6 消费,剩余商品: 0个
消费者 Thread-7 等待,剩余商品: 0个
消费者 Thread-8 等待,剩余商品: 0个
消费者 Thread-9 等待,剩余商品: 0个
生产者 Thread-1 生产,剩余商品: 1个
生产者 Thread-2 生产,剩余商品: 2个
消费者 Thread-5 消费,剩余商品: 1个
生产者 Thread-3 生产,剩余商品: 2个
消费者 Thread-6 消费,剩余商品: 1个
消费者 Thread-4 消费,剩余商品: 0个
生产者 Thread-1 生产,剩余商品: 1个
消费者 Thread-8 消费,剩余商品: 0个
消费者 Thread-7 等待,剩余商品: 0个
生产者 Thread-2 生产,剩余商品: 1个
消费者 Thread-9 消费,剩余商品: 0个
消费者 Thread-5 等待,剩余商品: 0个
生产者 Thread-3 生产,剩余商品: 1个
消费者 Thread-4 消费,剩余商品: 0个
消费者 Thread-6 等待,剩余商品: 0个
消费者 Thread-8 等待,剩余商品: 0个
生产者 Thread-2 生产,剩余商品: 1个

 

另外:

       Condition对象的构造函数可以接受一个Lock/RLock对象作为参数,如果没有指定,则Condition对象会在内部自行创建一个RLock;

       除了notify方法外,Condition对象还提供了notifyAll方法,可以通知waiting池中的所有线程尝试acquire内部锁。

       由于上述机制,处于waiting状态的线程只能通过notify方法唤醒,所以notifyAll的作用在于防止有线程永远处于沉默状态。

 

相关阅读

《Python多线程 threading模块[1] 概述》

《Python多线程 threading模块[2] 两种方式启动线程》

《Python多线程 threading模块[3] Thread类的重要函数》

《Python多线程 threading模块[4] Lock互斥锁》

《Python多线程 threading模块[5] 死锁的形成》

《Python多线程 threading模块[6] 可重入锁RLock》

《Python多线程 threading模块[7] Event线程通信》

《Python多线程 threading模块[8] Condition实现复杂同步》

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值