threading模块学习

最近在celery源码学习中经常看到event库的使用,想结合源码学习,重温threading.event()的使用

threading.Event 是 Python 标准库 threading 模块中的一个线程同步原语,用于线程间的等待与通知。它允许一个或多个线程等待某个"事件"发生(即事件被 set),而其它线程可以在合适的时机触发该事件。常用于线程间通信、同步、生产者-消费者等场景。


主要特性

  • 事件标志:内部维护一个布尔标志,初始为 False。
  • set():将事件标志设为 True,唤醒所有等待的线程。
  • clear():将事件标志设为 False,等待的线程会阻塞。
  • wait(timeout=None):阻塞当前线程,直到事件标志为 True 或超时。
  • is_set():查询事件标志当前状态。

1. 基本等待与通知

python 复制代码
import threading
import time

event = threading.Event()

def worker():
    print("Worker: 等待事件...")
    event.wait()
    print("Worker: 事件已触发,开始工作!")

t = threading.Thread(target=worker)
t.start()

time.sleep(2)
print("主线程: 2秒后触发事件")
event.set()
t.join()

输出:

makefile 复制代码
Worker: 等待事件...
主线程: 2秒后触发事件
Worker: 事件已触发,开始工作!

2. 多线程同时等待同一个事件

python 复制代码
import threading
import time

event = threading.Event()

def worker(n):
    print(f"Worker {n}: 等待事件...")
    event.wait()
    print(f"Worker {n}: 事件已触发,开始工作!")

threads = [threading.Thread(target=worker, args=(i,)) for i in range(3)]
for t in threads: t.start()

time.sleep(1)
print("主线程: 1秒后触发事件")
event.set()
for t in threads: t.join()

输出:

yaml 复制代码
Worker 0: 等待事件...
Worker 1: 等待事件...
Worker 2: 等待事件...
主线程: 1秒后触发事件
Worker 0: 事件已触发,开始工作!
Worker 1: 事件已触发,开始工作!
Worker 2: 事件已触发,开始工作!

3. 超时等待

python 复制代码
import threading
import time

event = threading.Event()

def worker():
    print("Worker: 最多等待3秒...")
    flag = event.wait(timeout=3)
    if flag:
        print("Worker: 事件已触发!")
    else:
        print("Worker: 超时未触发事件,继续执行...")

t = threading.Thread(target=worker)
t.start()

time.sleep(5)
event.set()
t.join()

输出:

makefile 复制代码
Worker: 最多等待3秒...
Worker: 超时未触发事件,继续执行...

4. 多次触发与清除

python 复制代码
import threading
import time

event = threading.Event()

def worker():
    while True:
        print("Worker: 等待事件...")
        event.wait()
        print("Worker: 事件已触发,工作一次!")
        event.clear()  # 清除事件,进入下一轮等待

t = threading.Thread(target=worker, daemon=True)
t.start()

for i in range(3):
    time.sleep(1)
    print(f"主线程: 第{i+1}次触发事件")
    event.set()

time.sleep(2)

输出:

makefile 复制代码
Worker: 等待事件...
主线程: 第1次触发事件
Worker: 事件已触发,工作一次!
Worker: 等待事件...
主线程: 第2次触发事件
Worker: 事件已触发,工作一次!
Worker: 等待事件...
主线程: 第3次触发事件
Worker: 事件已触发,工作一次!
Worker: 等待事件...

5. 生产者-消费者模型(事件控制)

python 复制代码
import threading
import time

event = threading.Event()
data = []

def producer():
    for i in range(5):
        time.sleep(1)
        data.append(i)
        print(f"Producer: 生产数据 {i}")
        event.set()  # 通知消费者有新数据
        event.clear()

def consumer():
    for _ in range(5):
        print("Consumer: 等待新数据...")
        event.wait()
        print(f"Consumer: 消费数据 {data[-1]}")

t1 = threading.Thread(target=producer)
t2 = threading.Thread(target=consumer)
t1.start()
t2.start()
t1.join()
t2.join()

输出:

makefile 复制代码
Consumer: 等待新数据...
Producer: 生产数据 0
Consumer: 消费数据 0
Consumer: 等待新数据...
Producer: 生产数据 1
Consumer: 消费数据 1
Consumer: 等待新数据...
Producer: 生产数据 2
Consumer: 消费数据 2
Consumer: 等待新数据...
Producer: 生产数据 3
Consumer: 消费数据 3
Consumer: 等待新数据...
Producer: 生产数据 4
Consumer: 消费数据 4

总结

  • threading.Event 适合多线程同步、等待、通知、超时等场景。
  • 通过 set/clear/wait/is_set 方法实现灵活的线程间通信。
  • 可用于生产者-消费者、批量唤醒、超时控制等多种并发模式。
相关推荐
mit6.8243 小时前
[VoiceRAG] Azure | 使用`azd`部署应用 | Dockerfile
python
砥锋3 小时前
计算机人的雷达入门:零基础用Python+Cinrad可视化雷达数据【实战指南】
python
你们瞎搞4 小时前
arcgis矢量数据转为标准geojson格式
python·arcgis·json·地理空间数据
郝学胜-神的一滴4 小时前
Python中的鸭子类型:理解动态类型的力量
开发语言·python·程序人生·软件工程
2401_841495644 小时前
【计算机视觉】霍夫变换函数的参数调整
人工智能·python·算法·计算机视觉·霍夫变换·直线检测·调整策略
猫头虎4 小时前
如何解决 pip install -r requirements.txt extras 语法 ‘package[extra’ 缺少 ‘]’ 解析失败问题
开发语言·python·开源·beautifulsoup·virtualenv·pandas·pip
eqwaak05 小时前
动态图表导出与视频生成:精通Matplotlib Animation与FFmpeg
开发语言·python·ffmpeg·音视频·matplotlib
AndrewHZ5 小时前
【图像处理基石】GIS图像处理入门:4个核心算法与Python实现(附完整代码)
图像处理·python·算法·计算机视觉·gis·cv·地理信息系统
帮帮志6 小时前
目录【系列文章目录】-(关于帮帮志,关于作者)
java·开发语言·python·链表·交互