十二.事件 (Event)(了解)
线程之间状态同步, 两个不同的任务执行, 一个任务如果需要另一个任务执行之后才能开始执行, 那么这个待执行的任务是如何获取到上一个任务执行状态的呢? 如果是进程, 那就需要借助共享内存传递一个标志信号, 而线程本身就共享一个线程的内存空间, 所以, 为了解决以上问题, threading 模块为我们提供了一个 Event 对象
1.Event 对象的方法
Event 本质就是一个标志, True或者False, 而它提供的wait函数可以阻塞当前线程, 直到状态从False变为True
- 导入方法 :
from threading import Event
方法 | 作用 |
---|---|
event.isSet( ) | 返回event的状态 |
event.wait( ) | 状态值为False为阻塞, 默认False |
event.set( ) | 设置event的状态值为True,所有阻塞池的线程激活进入就绪状态, 等待操作系统调度 |
event.clear( ) | 恢复event的状态值为False |
2.红绿灯示例
- 红灯停(False), 绿灯行(True)
from threading import Event,Thread,current_thread
import time
import random
e = Event() # 默认False
def f1(): # 红绿灯机制
while 1:
e.clear() # 先将其置为False
print("红灯亮")
time.sleep(3) # 模拟红灯等待三秒
e.set() # 将其设置为True
print("绿灯亮")
time.sleep(2) # 模拟绿灯通行两秒
def f2(): # 模拟行人
while 1:
if e.is_set(): # 判断如果是True,则通行
print(f"{current_thread().name}行人正在通行")
break # 行人走后结束这个线程
else:
print(f"{current_thread().name}正在等待")
e.wait() # 正在阻塞状态,当event变为True时就激活
if __name__ == '__main__':
Thread(target=f1).start() # 创建一个红绿灯线程
while 1:
time.sleep(random.randint(0,2))
Thread(target=f2).start() # 随机时间产生一个行人
'''输出
红灯亮
Thread-2正在等待
Thread-3正在等待
Thread-4正在等待
绿灯亮
Thread-2行人正在通行
Thread-4行人正在通行
Thread-3行人正在通行
Thread-5行人正在通行
红灯亮
Thread-6正在等待
绿灯亮
Thread-6行人正在通行
......
....
```
十三.定时器 (Timer)
指定 n 秒后执行操作
- 用法 : Timer(float,function,args=(,))
- 参数 : 指定时间、函数名、函数参数
from threading import Timer
import time
def test(name):
print(f"{name}黑猫警长,用时:{time.time()-start_time}")
start_time = time.time()
p = Timer(3.5,test,args=("葫芦娃",))
p.start()
'''输出
葫芦娃黑猫警长,用时:3.5001583099365234
'''
十四.线程Queue
队列主要用于进程与进程之间的通信, 因为它们内存空间不共享, 线程之间数据是共享的, 使用队列是因为队列的实现原理是管道+锁, 它能保证数据的安全性
ps : 目前我们使用的队列都只能单机测试使用, 以后我们使用的是基于网络的别人封装好了的队列框架
用法 : import queue
, 使用方式与进程queue无异, 以下介绍 queue 模块下三种类的使用
1.先进先出 (FIFO) 队列 Queue
import queue
q = queue.Queue(3)
q.put("shawn")
q.put(123)
q.put([1,2,3,4])
try:
q.put({"name":"shawn"},block=False)
except Exception:
print("放满了--->")
print(q.get())
print(q.get())
print(q.get())
try:
print(q.get())
except Exception:
print("取完了--->")
'''输出
放满了
shawn
123
[1, 2, 3, 4]
取完了
'''
2.后进先出 (LIFO) 堆栈 LifoQueue
import queue
q = queue.LifoQueue(3)
q.put("shawn")
q.put(123)
q.put([1,2,3,4])
print(q.get())
print(q.get())
print(q.get())
'''输出
[1, 2, 3, 4]
123
shawn
'''
3.优先级队列 PriorityQueue
- 用法 :
put(([等级],[数据]))
- 等级小的优先级越高
import queue
q = queue.PriorityQueue(3)
q.put((45,"shawn"))
q.put((10,123))
q.put((-20,[1,2,3,4]))
print(q.get())
print(q.get())
print(q.get())
'''输出
(-20, [1, 2, 3, 4])
(10, 123)
(45, 'shawn')
'''