- 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 - 元组(Tuple )
- Python - 访问元组项
- Python - 更新元组
- Python - 解压缩元组项
- Python - 循环元组
- Python - 联接元组
- Python - 元组方法
- Python - 元组练习
- Python 集
- Python - 集(sets)
- Python - 访问 Set Items
- Python - 添加 Set Items
- Python - 删除 Set Items
- Python - 循环 Set Items
- Python - 联接 Sets
- Python - 复制 Set
- Python - Set 运算符
- Python - Set 方法
- Python - Set 的练习
- 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 文件/目录方法
- 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 - 异常处理
- 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 - 日期和时间
- Python - math 模块
- Python - 迭代器
- Python - 生成器
- Python - 闭包(closures)
- Python - 装饰器( Decorators)
- Python - 递归
- Python - 正则表达式
- Python - PIP
- Python - 数据库访问
- Python - 弱引用
- Python - 序列化
- Python - 模板
- Python - 输出格式
- Python - 性能测量
- Python - 数据压缩
- Python - CGI 编程
- Python - XML 处理
- Python - GUI 编程
- Python - 命令行参数
- Python - 文档字符串
- Python - JSON
- Python - 发送电子邮件
- Python - 更多扩展
- Python - 工具/实用程序
- Python - 图形用户界面
- Python 高级概念
- Python - 抽象基类
- Python - 自定义异常
- Python - 高阶函数
- Python - 对象内部
- Python - 内存管理
- Python - 元类
- Python - 使用 Metaclasses 进行元编程
- Python - 模拟和存根
- Python - 猴子修补
- Python - 信号处理
- Python - 类型提示
- Python - 自动化教程
- Python - 人性化软件包
- Python - 上下文管理器
- Python - 协程
- Python - 描述符
- Python - 诊断和修复内存泄漏
- Python - 不可变数据结构
Python - 线程死锁
死锁可以描述为并发失败模式。在程序中,一个或多个线程等待从未发生的条件的情况。因此,线程无法继续,程序卡住或冻结,必须手动终止。
在并发程序中,死锁情况可能以多种方式出现。死锁从来都不是有意开发的,相反,它们实际上是代码中的副作用或 bug。
下面列出了线程死锁的常见原因 -
- 尝试两次获取同一互斥锁的线程。
- 彼此等待的线程(例如 A 等待 B,B 等待 A)。
- 当线程无法释放 lock、semaphore、condition、event 等资源时。
- 以不同顺序获取互斥锁的线程(例如,无法执行锁排序)。
如何避免 Python 线程中的死锁
当多线程应用程序中的多个线程尝试访问同一资源时,例如对同一文件执行读/写操作,可能会导致数据不一致。因此,使用锁定机制同步对资源的并发访问非常重要。
Python threading 模块提供了一种易于实现的锁定机制来同步线程。您可以通过调用 Lock() 类来创建新的锁对象,该类将锁初始化为解锁状态。
带有 Lock 对象的锁定机构
Lock 类的对象有两种可能的状态 - 锁定或解锁,最初在首次创建时处于解锁状态。锁不属于任何特定线程。
Lock 类定义 acquire() 和 release() 方法。
acquire() 方法
Lock 类的 acquire() 方法将锁的状态从 unlocked 更改为 locked。除非可选的 blocking 参数设置为 True,否则它会立即返回,在这种情况下,它会等待直到获取锁。
这是此方法的语法 -
Lock.acquire(blocking, timeout)
哪里
- blocking − 如果设置为 False,则表示不阻止。如果阻止设置为 True 的调用会阻止,则立即返回 False;否则,将锁设置为 locked 并返回 True。
- timeout - 指定获取锁的超时期限。
如果成功获取锁,则此方法的返回值为 True;否则为 False。
release() 方法
当状态为 locked 时,另一个线程中的此方法会将其更改为 unlocked。这可以从任何线程调用,而不仅仅是从已获取锁的线程调用
以下是 release() 方法的语法 -
Lock.release()
release() 方法只能在 locked 状态下调用。如果尝试释放未锁定的锁,将引发 RuntimeError。
锁定后,将其重置为解锁状态,然后返回。如果任何其他线程被阻塞,等待锁解锁,则只允许其中一个线程继续。此方法没有返回值。
例在下面的程序中,两个线程尝试调用 synchronized() 方法。其中一个服务器获取锁并获得访问权限,而另一个服务器等待。当第一个线程的 run() 方法完成时,锁将被释放,并且 synchronized 方法可用于第二个线程。
当两个线程都加入时,程序结束。
from threading import Thread, Lock
import time
lock=Lock()
threads=[]
class myThread(Thread):
def __init__(self,name):
Thread.__init__(self)
self.name=name
def run(self):
lock.acquire()
synchronized(self.name)
lock.release()
def synchronized(threadName):
print ("{} has acquired lock and is running synchronized method".format(threadName))
counter=5
while counter:
print ('**', end='')
time.sleep(2)
counter=counter-1
print('\nlock released for', threadName)
t1=myThread('Thread1')
t2=myThread('Thread2')
t1.start()
threads.append(t1)
t2.start()
threads.append(t2)
for t in threads:
t.join()
print ("end of main thread")
它将产生以下输出 -
**********
lock released for Thread1
Thread2 has acquired lock and is running synchronized method
**********
lock released for Thread2
end of main thread
用于同步的信号量对象
除了锁之外,Python 线程模块还支持信号量,这提供了另一种同步技术。它是著名计算机科学家 Edsger W. Dijkstra 发明的最古老的同步技术之一。
信号量的基本概念是使用内部计数器,该计数器由每个 acquire() 调用递减,并由每个 release() 调用递增。计数器永远不会低于零;当 acquire() 发现它为零时,它会阻塞,直到其他线程调用 release()。
threading 模块中的 Semaphore 类定义了 acquire() 和 release() 方法。
acquire() 方法
如果内部计数器在输入时大于零,则将其减量 1 并立即返回 True。
如果内部计数器在输入时为零,则阻塞直到被调用 release() 唤醒。唤醒 (且计数器大于 0) 后,将计数器递减 1 并返回 True。每次调用 release() 都会唤醒一个线程。线程唤醒的顺序是任意的。
如果 blocking 参数设置为 False,则不阻止。如果没有参数的调用会阻塞,则立即返回 False;否则,执行与不带参数调用时相同的操作,并返回 True。
release() 方法
释放信号量,将内部计数器增加 1。当它在输入时为零,并且其他线程正在等待它再次大于零时,唤醒其中的 n 个线程。
例
此示例演示如何在 Python 中使用 Semaphore 对象来控制对多个线程之间共享资源的访问,以避免 Python 多线程程序中的死锁。
from threading import *
import time
# creating thread instance where count = 3
lock = Semaphore(4)
# creating instance
def synchronized(name):
# calling acquire method
lock.acquire()
for n in range(3):
print('Hello! ', end = '')
time.sleep(1)
print( name)
# calling release method
lock.release()
# creating multiple thread
thread_1 = Thread(target = synchronized , args = ('Thread 1',))
thread_2 = Thread(target = synchronized , args = ('Thread 2',))
thread_3 = Thread(target = synchronized , args = ('Thread 3',))
# calling the threads
thread_1.start()
thread_2.start()
thread_3.start()
它将产生以下输出 -
Hello! Thread 2
Thread 3
Hello! Hello! Thread 1
Hello! Thread 3
Thread 2
Hello! Hello! Thread 1
Thread 3
Thread 2