主页
  • 主页
  • 分类
  • 热文
  • 教程
  • 面试
  • 标签
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 死锁


上一章 下一章

死锁是一种程序中的情况,其中一个或多个线程等待一个永远不会发生的条件。结果,这些线程无法进展,程序停滞或冻结,必须手动终止。

死锁可能以多种方式出现在您的并发程序中。死锁从来不是有意开发的,而是代码中的副作用或错误。

导致线程死锁的常见原因如下:

  • 一个线程尝试两次获取同一个互斥锁。

  • 线程相互等待(例如,A 等待 B,B 等待 A)。

  • 当线程未能释放资源,如锁、信号量、条件、事件等。

  • 线程以不同顺序获取互斥锁(例如,未能执行锁排序)。

如何避免 Python 线程中的死锁

当多线程应用程序中的多个线程尝试访问相同的资源,比如对同一个文件进行读写操作时,可能导致数据不一致。因此,重要的是使用锁定机制来同步对资源的并发访问。

Python 的 threading 模块提供了一种易于实现的锁定机制来同步线程。您可以通过调用 Lock() 类创建一个新的锁对象,初始化时锁处于未锁定状态。

使用 Lock 对象的锁定机制

Lock 类的对象有两种可能的状态——锁定或未锁定,初始创建时为未锁定状态。一个锁不属于任何特定的线程。

Lock 类定义了 acquire() 和 release() 方法。

acquire() 方法

Lock 类的 acquire() 方法将锁的状态从未锁定改为锁定。除非可选的 blocking 参数设置为 True,否则它会立即返回。在这种情况下,它会等待直到锁被获取。

以下是该方法的语法:

Lock.acquire(blocking, timeout)

其中,

  • blocking - 如果设置为 False,意味着不要阻塞。如果带有 blocking 设置为 True 的调用将会阻塞,则立即返回 False;否则,将锁设置为锁定并返回 True。

  • timeout - 指定获取锁的超时时间。

该方法如果成功获取锁则返回值为 True;如果不成功则为 False。

release() 方法

当状态为锁定时,此方法在另一个线程中将其改为未锁定。这可以从任何线程调用,而不仅仅是获取锁的线程。

以下是 release() 方法的语法:

Lock.release()

release() 方法应该只在锁定状态下调用。如果尝试释放未锁定的锁,将引发 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("{} 已经获取锁并正在运行同步方法".format(threadName))
    counter = 5
    while counter:
        print('**', end='')
        time.sleep(2)
        counter -= 1
    print('\n锁已被释放', threadName)

t1 = myThread('Thread1')
t2 = myThread('Thread2')

t1.start()
threads.append(t1)

t2.start()
threads.append(t2)

for t in threads:
    t.join()
print("主线程结束")

它将产生以下输出:

Thread1 已经获取锁并正在运行同步方法
**********
锁已被释放 Thread1
Thread2 已经获取锁并正在运行同步方法
**********
锁已被释放 Thread2
主线程结束

使用 Semaphore 对象进行同步

除了锁之外,Python 的 threading 模块还支持信号量,这是由著名计算机科学家 Edsger W. Dijkstra 发明的最古老的同步技术之一。

信号量的基本概念是使用一个内部计数器,每次调用 acquire() 时递减,每次调用 release() 时递增。计数器永远不能低于零;当 acquire() 发现它是零时,它将阻塞,直到其他线程调用 release()。

threading 模块中的 Semaphore 类定义了 acquire() 和 release() 方法。

acquire() 方法

如果进入时内部计数器大于零,则递减计数器并立即返回 True。

如果进入时内部计数器为零,则阻塞直到被 release() 调用唤醒。一旦被唤醒(且计数器大于零),递减计数器并返回 True。每次调用 release() 恰好唤醒一个线程。线程唤醒的顺序是任意的。

如果 blocking 参数设置为 False,则不要阻塞。如果无参数调用将会阻塞,则立即返回 False;否则,做同样的事情并返回 True。

release() 方法

释放信号量,将内部计数器递增 1。当它进入时为零且其他线程正等待它再次大于零时,唤醒 n 个这样的线程。

示例

此示例演示了如何在 Python 中使用 Semaphore 对象来控制多个线程之间对共享资源的访问,从而避免 Python 多线程程序中的死锁。

from threading import *
import time

# 创建 Semaphore 实例,count = 3
lock = Semaphore(4)

# 创建实例
def synchronized(name):
   
   # 调用 acquire 方法
   lock.acquire()

   for n in range(3):
      print('Hello! ', end='')
      time.sleep(1)
      print(name)

      # 调用 release 方法
      lock.release()

# 创建多个线程
thread_1 = Thread(target=synchronized, args=('Thread 1',))
thread_2 = Thread(target=synchronized, args=('Thread 2',))
thread_3 = Thread(target=synchronized, args=('Thread 3',))

# 调用线程
thread_1.start()
thread_2.start()
thread_3.start()

它将产生以下输出:

Hello! Hello! Hello! Thread 1
Hello! Thread 2
Thread 3
Hello! Hello! Thread 1
Hello! Thread 3
Thread 2
Hello! Hello! Thread 1
Thread 3
Thread 2
上一章 下一章
阅读号二维码

关注阅读号

联系二维码

联系我们

© 2024 Yoagoa. All rights reserved.

粤ICP备18007391号

站点地图