主页
  • 主页
  • 分类
  • 热文
  • 教程
  • 面试
  • 标签
Python

Python 基础

Python 主页
Python 概述
Python 历史
Python 功能
Python 与 C++
Python Hello World
Python 应用领域
Python 解释器及其模式
Python 环境设置
Python 虚拟环境
Python 语法
Python 变量
Python 数据类型
Python 类型转换
Python Unicode 系统
Python 文字
Python 运算符
Python 算术运算符
Python 比较运算符
Python 赋值运算符
Python 逻辑运算符
Python 位运算符
Python 成员运算符
Python 身份运算符
Python 运算符优先级
Python 注释
Python 用户输入
Python 数字
Python 布尔值

Python 控制语句

Python 控制流
Python 决策
Python if 语句
Python if-else 语句
Python 嵌套 if 语句
Python match-case 语句
Python 循环
Python For 循环
Python for-else 循环
Python while 循环
Python break 语句
Python Continue 语句
Python pass 语句
Python 嵌套循环

Python 函数和模块

Python 函数
Python 默认参数
Python 关键字参数
Python 关键字专用参数
Python 位置参数
Python 仅限位置参数
Python 任意或可变长度参数
Python 变量作用域
Python 函数注释
Python 模块
Python 内置函数

Python 字符串

Python 字符串
Python 字符串切片
Python 字符串修改
Python 字符串连接
Python 字符串格式化
Python 转义字符
Python 字符串方法
Python 字符串练习

Python 列表

Python 列表
Python 访问列表项
Python 更改列表项
Python 添加列表项
Python 删除列表项
Python 循环列表
Python 列表推导式
Python 排序列表
Python 复制列表
Python 合并列表
Python 列表方法
Python 列表练习

Python 元组

Python 元组
Python 访问元组项
Python 更新元组
Python 解包元组项
Python 循环元组
Python 合并元组
Python 元组方法
Python 元组练习

Python 集合

Python 集合
Python 访问集合项
Python 添加集合项
Python 删除集合项
Python 循环集合
Python 合并集合
Python 复制集合
Python 集合运算符
Python 集合方法
Python 集合练习

Python 字典

Python 字典
Python 访问字典项
Python 更改字典项
Python 添加字典项
Python 移除字典项
Python 字典视图对象
Python 循环字典
Python 复制字典
Python 嵌套字典
Python 字典方法
Python 字典练习

Python 数组

Python 数组
Python 访问数组项
Python 添加数组项
Python 移除数组项
Python 循环数组
Python 复制数组
Python 反转数组
Python 排序数组
Python 合并数组
Python 数组方法
Python 数组练习

Python 文件处理

Python 文件处理
Python 文件写入
Python 文件读取
Python 重命名和删除文件
Python 目录
Python 文件方法
Python 文件/目录方法
Python OS.Path 方法

Python 面向对象编程

Python OOP 概念
Python 类和对象
Python 类属性
Python 类方法
Python 静态方法
Python 构造函数
Python 访问修饰符
Python 继承
Python 多态
Python 方法重写
Python 方法重载
Python 动态绑定
Python 动态类型
Python 抽象
Python 封装
Python 接口
Python 包
Python 内部类
Python 匿名类和对象
Python 单例类
Python 包装器类
Python 枚举
Python 反射

Python 错误和异常

Python 语法错误
Python 异常处理
Python Try-Except
Python Try-Finally
Python 抛出异常
Python 异常链
Python 嵌套 try
Python 用户定义异常
Python 日志记录
Python 断言
Python 内置异常

Python 多线程

Python 多线程
Python 线程生命周期
Python 创建线程
Python 启动线程
Python 合并线程
Python 命名线程
Python 线程调度
Python 线程池
Python 主线程
Python 线程优先级
Python 守护线程
Python 线程同步

Python 同步

Python 线程间通信
Python 死锁
Python 中断线程

Python 网络

Python 网络编程
Python 套接字编程
Python URL 处理
Python 泛型

Python 杂项

Python Date and Time
Python math模块
Python 迭代器
Python 生成器
Python 闭包
Python 装饰器
Python 递归
Python 正则表达式
Python Pip
Python 数据库访问
Python 弱引用
Python 序列化
Python 模板技术
Python 输出格式化
Python 性能测量
Python 数据压缩
Python 通用网关接口
Python XML 处理
Python 用户界面(GUI)
Python 命令行参数
Python Docstrings
Python JSON
Python 发送电子邮件
Python 进一步扩展
Python 工具/实用程序
Python GUI

Python 高级概念

Python 抽象基类
Python 自定义异常
Python 高阶函数
Python 对象的内部机制
Python 内存管理
Python 元类
Python 元编程
Python 模拟与桩
Python 猴子补丁
Python 信号处理
Python 类型提示
Python 进行自动化
Python Humanize包
Python 上下文管理器
Python 协程
Python 描述符
Python 内存泄漏
Python 不可变数据结构

基础

Python 主页
Python 概述
Python 历史
Python 功能
Python 与 C++
Python Hello World
Python 应用领域
Python 解释器及其模式
Python 环境设置
Python 虚拟环境
Python 语法
Python 变量
Python 数据类型
Python 类型转换
Python Unicode 系统
Python 文字
Python 运算符
Python 算术运算符
Python 比较运算符
Python 赋值运算符
Python 逻辑运算符
Python 位运算符
Python 成员运算符
Python 身份运算符
Python 运算符优先级
Python 注释
Python 用户输入
Python 数字
Python 布尔值

控制语句

Python 控制流
Python 决策
Python if 语句
Python if-else 语句
Python 嵌套 if 语句
Python match-case 语句
Python 循环
Python For 循环
Python for-else 循环
Python while 循环
Python break 语句
Python Continue 语句
Python pass 语句
Python 嵌套循环

函数和模块

Python 函数
Python 默认参数
Python 关键字参数
Python 关键字专用参数
Python 位置参数
Python 仅限位置参数
Python 任意或可变长度参数
Python 变量作用域
Python 函数注释
Python 模块
Python 内置函数

字符串

Python 字符串
Python 字符串切片
Python 字符串修改
Python 字符串连接
Python 字符串格式化
Python 转义字符
Python 字符串方法
Python 字符串练习

列表

Python 列表
Python 访问列表项
Python 更改列表项
Python 添加列表项
Python 删除列表项
Python 循环列表
Python 列表推导式
Python 排序列表
Python 复制列表
Python 合并列表
Python 列表方法
Python 列表练习

元组

Python 元组
Python 访问元组项
Python 更新元组
Python 解包元组项
Python 循环元组
Python 合并元组
Python 元组方法
Python 元组练习

集合

Python 集合
Python 访问集合项
Python 添加集合项
Python 删除集合项
Python 循环集合
Python 合并集合
Python 复制集合
Python 集合运算符
Python 集合方法
Python 集合练习

字典

Python 字典
Python 访问字典项
Python 更改字典项
Python 添加字典项
Python 移除字典项
Python 字典视图对象
Python 循环字典
Python 复制字典
Python 嵌套字典
Python 字典方法
Python 字典练习

数组

Python 数组
Python 访问数组项
Python 添加数组项
Python 移除数组项
Python 循环数组
Python 复制数组
Python 反转数组
Python 排序数组
Python 合并数组
Python 数组方法
Python 数组练习

文件处理

Python 文件处理
Python 文件写入
Python 文件读取
Python 重命名和删除文件
Python 目录
Python 文件方法
Python 文件/目录方法
Python OS.Path 方法

面向对象编程

Python OOP 概念
Python 类和对象
Python 类属性
Python 类方法
Python 静态方法
Python 构造函数
Python 访问修饰符
Python 继承
Python 多态
Python 方法重写
Python 方法重载
Python 动态绑定
Python 动态类型
Python 抽象
Python 封装
Python 接口
Python 包
Python 内部类
Python 匿名类和对象
Python 单例类
Python 包装器类
Python 枚举
Python 反射

错误和异常

Python 语法错误
Python 异常处理
Python Try-Except
Python Try-Finally
Python 抛出异常
Python 异常链
Python 嵌套 try
Python 用户定义异常
Python 日志记录
Python 断言
Python 内置异常

多线程

Python 多线程
Python 线程生命周期
Python 创建线程
Python 启动线程
Python 合并线程
Python 命名线程
Python 线程调度
Python 线程池
Python 主线程
Python 线程优先级
Python 守护线程
Python 线程同步

同步

Python 线程间通信
Python 死锁
Python 中断线程

网络

Python 网络编程
Python 套接字编程
Python URL 处理
Python 泛型

杂项

Python Date and Time
Python math模块
Python 迭代器
Python 生成器
Python 闭包
Python 装饰器
Python 递归
Python 正则表达式
Python Pip
Python 数据库访问
Python 弱引用
Python 序列化
Python 模板技术
Python 输出格式化
Python 性能测量
Python 数据压缩
Python 通用网关接口
Python XML 处理
Python 用户界面(GUI)
Python 命令行参数
Python Docstrings
Python JSON
Python 发送电子邮件
Python 进一步扩展
Python 工具/实用程序
Python GUI

高级概念

Python 抽象基类
Python 自定义异常
Python 高阶函数
Python 对象的内部机制
Python 内存管理
Python 元类
Python 元编程
Python 模拟与桩
Python 猴子补丁
Python 信号处理
Python 类型提示
Python 进行自动化
Python Humanize包
Python 上下文管理器
Python 协程
Python 描述符
Python 内存泄漏
Python 不可变数据结构

Python 线程间通信


上一章 下一章

线程间通信指的是在 Python 多线程程序中启用线程之间的通信和同步的过程。

通常情况下,Python 中的线程在同一进程的内存空间内共享,这使得它们能够通过共享变量、对象以及 threading 模块提供的专门同步机制来交换数据和协调活动。

为了促进线程间的通信,threading 模块提供了各种同步原语,如 Locks、Events、Conditions 和 Semaphores 对象。在本教程中,您将学习如何使用 Event 和 Condition 对象来在多线程程序中提供线程间的通信。

Event 对象

Event 对象管理一个内部标志的状态,以便线程可以等待或设置。Event 对象提供了用于控制此标志状态的方法,允许线程基于共享条件来同步它们的活动。

该标志最初为假,在使用 set() 方法后变为真,并在使用 clear() 方法后重置为假。wait() 方法会在标志为真时阻塞。

以下为 Event 对象的关键方法:

  • is_set(): 如果内部标志为真则返回 True。
  • set(): 将内部标志设置为真。所有等待标志变为真的线程将被唤醒。在标志为真之后调用 wait() 的线程将不会被阻塞。
  • clear(): 将内部标志重置为假。随后调用 wait() 的线程将阻塞,直到 set() 被调用再次将内部标志设置为真。
  • wait(timeout=None): 阻塞直到内部标志为真。如果进入时内部标志为真,则立即返回。否则,阻塞直到另一线程调用 set() 将标志设置为真,或者直到可选的超时发生。当存在超时参数并且不为 None 时,它应为指定操作超时秒数的浮点数。

示例

以下代码试图模拟交通流量由红绿灯状态控制的情况。

程序中有两个线程,分别针对两个不同的函数。signal_state() 函数周期性地设置和重置事件,指示信号从绿变红。

traffic_flow() 函数等待事件被设置,并在事件保持设置的情况下运行一个循环。

from threading import Event, Thread
import time

terminate = False

def signal_state():
    global terminate
    while not terminate:
        time.sleep(0.5)
        print("交通警察给出绿灯信号")
        event.set()
        time.sleep(1)
        print("交通警察给出红灯信号")
        event.clear()

def traffic_flow():
    global terminate
    num = 0
    while num < 10 and not terminate:
        print("等待绿灯信号")
        event.wait()
        print("绿灯信号... 交通可以移动")
        while event.is_set() and not terminate:
            num += 1
            print("车辆编号:", num," 通过信号灯")
            time.sleep(1)
        print("红灯信号... 交通需要等待")

event = Event()
t1 = Thread(target=signal_state)
t2 = Thread(target=traffic_flow)
t1.start()
t2.start()

# 在一段时间后终止线程
time.sleep(5)
terminate = True

# join 所有线程以完成
t1.join()
t2.join()

print("退出主线程")

输出:

等待绿灯信号
交通警察给出绿灯信号
绿灯信号... 交通可以移动
车辆编号: 1  通过信号灯
交通警察给出红灯信号
红灯信号... 交通需要等待
等待绿灯信号
交通警察给出绿灯信号
绿灯信号... 交通可以移动
车辆编号: 2  通过信号灯
车辆编号: 3  通过信号灯
交通警察给出红灯信号
交通警察给出绿灯信号
车辆编号: 4  通过信号灯
交通警察给出红灯信号
红灯信号... 交通需要等待
交通警察给出绿灯信号
交通警察给出红灯信号
退出主线程

Condition 对象

Python 的 threading 模块中的 Condition 对象提供了一个更高级的同步机制。它允许线程等待另一个线程的通知再继续。Condition 对象总是与一个锁关联,并提供线程间信号的机制。

以下是 threading.Condition() 类的语法:

threading.Condition(lock=None)

以下是 Condition 对象的关键方法:

  • acquire(*args): 获取底层锁。此方法调用底层锁上的相应方法;返回值是该方法返回的值。
  • release(): 释放底层锁。此方法调用底层锁上的相应方法;没有返回值。
  • wait(timeout=None): 此方法释放底层锁,然后阻塞直到它被同一条件变量在另一线程中的 notify() 或 notify_all() 调用唤醒,或者直到可选的超时发生。一旦被唤醒或超时,它重新获取锁并返回。
  • wait_for(predicate, timeout=None): 此实用方法可能会反复调用 wait() 直到满足谓词,或者直到发生超时。返回值是谓词的最后一个返回值,并将在方法超时时评估为假。
  • notify(n=1): 此方法最多唤醒 n 个等待条件变量的线程;如果没有线程在等待,则此方法是空操作。
  • notify_all(): 唤醒所有等待此条件的线程。此方法的行为像 notify(),但是唤醒所有等待的线程而不是一个。如果调用此方法时调用线程未获取锁,则抛出 RuntimeError。

示例

此示例演示了使用 Python 的 threading 模块中的 Condition 对象进行线程间通信的简单形式。在此,thread_a 和 thread_b 通过一个 Condition 对象进行通信,thread_a 在接收到 thread_b 的通知之前处于等待状态。thread_b 在通知 thread_a 之前休眠两秒,然后完成。

from threading import Condition, Thread
import time

c = Condition()

def thread_a():
    print("线程 A 开始")
    with c:
        print("线程 A 等待许可...")
        c.wait()
        print("线程 A 获得许可!")
    print("线程 A 完成")

def thread_b():
    print("线程 B 开始")
    with c:
        time.sleep(2)
        print("通知线程 A...")
        c.notify()
    print("线程 B 完成")

Thread(target=thread_a).start()
Thread(target=thread_b).start()

输出:

线程 A 开始
线程 A 等待许可...
线程 B 开始
通知线程 A...
线程 B 完成
线程 A 获得许可!
线程 A 完成

示例

以下是另一个代码示例,演示了 Condition 对象是如何用于提供线程间的通信的。在此,线程 t2 运行 taskB() 函数,而线程 t1 运行 taskA() 函数。线程 t1 获取条件并通知它。

此时,线程 t2 处于等待状态。在条件被释放后,等待的线程继续消费由通知函数生成的随机数。

from threading import Condition, Thread
import time
import random

numbers = []

def taskA(c):
    for _ in range(5):
        with c:
            num = random.randint(1, 10)
            print("生成随机数:", num)
            numbers.append(num)
            print("通知已发出")
            c.notify()
        time.sleep(0.3)

def taskB(c):
    for i in range(5):
        with c:
            print("等待更新")
            while not numbers: 
                c.wait()
            print("获得随机数", numbers.pop())
        time.sleep(0.3)

c = Condition()
t1 = Thread(target=taskB, args=(c,))
t2 = Thread(target=taskA, args=(c,))
t1.start()
t2.start()
t1.join()
t2.join()
print("完成")

当你执行这段代码时,它会产生如下输出:

等待更新
生成随机数: 2
通知已发出
获得随机数 2
生成随机数: 5
通知已发出
等待更新
获得随机数 5
生成随机数: 1
通知已发出
等待更新
获得随机数 1
生成随机数: 9
通知已发出
等待更新
获得随机数 9
生成随机数: 2
通知已发出
等待更新
获得随机数 2
完成
上一章 下一章
阅读号二维码

关注阅读号

联系二维码

联系我们

© 2024 Yoagoa. All rights reserved.

粤ICP备18007391号

站点地图