在上一篇同步举的例子中,我们模拟了向资源池中填入和消耗资源这个过程,填入和消耗的时间都是不确定的,其实我们用信号量已经实现了线程间的通信–在表示共享资源的数量上。不过python还给我们提供了queue模块(Python 2.x版本中为Queue),它提供了线程间的通信机制,可以让线程之间可以分享数据。

具体来说,实现的方式也是和很简单的,可以理解为创建了一个公共可以访问的队列,线程都可以访问它。具体的queue模块有的一些属性如下:

queue模块中的类:

  • Queue(maxsize=0):创建一个先入先出的队列,最大值可以限制队列的空间,如果没有限定则为无限值。
  • LifoQueue(maxsize=0):创建一个先入后出的队列,也就是类似栈的东西。最大值可以限制队列的空间,如果没有限定则为无限值。
  • PriorQueue(maxsize=0):创建一个优先级的队列,最大值可以限制队列的空间,如果没有限定则为无限值。

异常:

  • Empty():对列为空的时候调用get*()方法抛出异常
  • Full():队列满时调用put*()方法抛出异常

queue对象方法:

  • qsize():返回队列大小,不过其他线程可能改变队列大小,这个值不准确。
  • empty():判断队列是为空,布尔值空为True,否则Flase。
  • full():判断队列是否满,布尔值满为True,否则为Flase。
  • put(item, block=True, timeout=None):把item放入队列中,block=True,timeout=None,则会等待直到有可用空间,timeout有值则最多阻塞多少秒。block=Flase就会抛出Empty异常。
  • put_nowait(item)同put(item,Flase)。
  • get(block=True,timeout=None):从队列中取得元素,block=True则阻塞到有可用元素
  • get_nowait():同get(Flase)。
  • task_done():用于表示队列中某个元素执行完成,和下一个方法join()一起配合使用
  • join():在队列中所有元素执行完前并调用上一个task_done()之前保持阻塞。

之前的信号量例子,改用queue来进行资源的共享

from random import randint
from queue import Queue
from time import sleep
from test_8_25.mythread import MyThread


def write_queue(queue):
    # 添加队列中元素行为
    print('Producing object for queue...', queue.put('xxx', 1))
    print('Size now', queue.qsize())


def read_queue(queue):
    #消耗队列中元素行为
    queue.get(1)
    print('Consumed object from queue...')
    print('Size now', queue.qsize())


def writer(queue, loops):
    #随机向队列中添加随机个数的元素
    for i in range(loops):
        write_queue(queue)
        sleep(randint(1, 3))


def reader(queue, loops):
    # 随机消耗队列中随机个数的元素
    for i in range(loops):
        read_queue(queue)
        sleep(randint(2,5))#等待时间比writer常,尽量避免队列在消耗时为空


funcs = [writer, reader]
nfuncs = range(len(funcs))


def main():
    nloops =randint(2,5)#随机产生线程数
    q = Queue(32) #获得共享队列

    threads = [] #线程列表
    for i in nfuncs: #生成所有的线程列表
        t = MyThread(funcs[i], (q, nloops), funcs[i].__name__)
        threads.append(t)

    for i in nfuncs:
        threads[i].start() #开启所有的线程

    for i in nfuncs:
        threads[i].join() #这里需要将线程挂起,否则主线程会线程结束前打印“all done”
        # 或者你也可以使用atexit()来限制在主线程退出时在打印

    print("All done.")


if __name__ == '__main__':
    main()

结果实例(结果随机不尽相同):

Starting writer at: Sat Aug 26 19:27:42 2017
Producing object for queue... None
Size now 1
Starting reader at: Sat Aug 26 19:27:42 2017
Consumed object from queue...
Size now 0
Producing object for queue... None
Size now 1
Producing object for queue... None
Size now 2
Consumed object from queue...
Size now 1
Consumed object from queue...
Size now 0
Producing object for queue... None
Size now 1
Consumed object from queue...
Size now 0
writer finished at: Sat Aug 26 19:27:53 2017
reader finished at: Sat Aug 26 19:27:53 2017
All done.