Python教程:一文掌握Python多线程(很详细)
目录
1.什么是多线程?
1.1多线程与单线程的区别
1.2 Python 中的多线程实现方式
2.使用 threading 模块创建和管理线程
2.1创建线程:Thread 类的基本用法
2.2线程的启动和执行:start() 方法
2.3线程的同步和阻塞:join() 方法
2.4线程的名称和标识:name 和 ident 属性
3.线程间的通信与同步
3.1使用 Lock 实现线程同步
3.2使用 Queue 实现线程间通信
4.线程池的使用
5.理解全局解释器锁(GIL)对多线程的影响
5.1作用和原理:
5.2对多线程并发执行的限制:
6.多线程中的常见问题与解决方法
6.1死锁(Deadlock)的原因及避免方法:
6.2线程间数据共享与安全访问:
7.线程越多越好么?
8.面试问题:解析Python中的GIL(全局解释器锁)是什么?它如何影响多线程编程?
问题描述:
详细答案:
9.多线程实战示例
9.1项目说明:
9.2完整代码示例:
9.3代码说明:
1.什么是多线程?
多线程是指在同一进程内同时运行多个线程,每个线程执行不同的任务,实现并发执行。每个线程都有自己的执行路径,可以独立运行和调度,共享进程的资源。
1.1多线程与单线程的区别
-
单线程: 指在程序中只有一个执行线程,按照顺序依次执行任务。单线程模型简单直观,但无法充分利用多核处理器的性能。
-
多线程: 可以同时执行多个线程,提高程序的响应速度和效率。多线程模型适用于需要同时处理多个任务或需要利用多核处理器的场景。
1.2 Python 中的多线程实现方式
Python 提供了 threading 模块来支持多线程编程。通过创建 Thread 对象并指定目标函数,可以实现线程的创建和管理。以下是一个简单的示例:
import threadingdef print_numbers():for i in range(1, 6):print(i)# 创建线程
t = threading.Thread(target=print_numbers)# 启动线程
t.start()# 主线程继续执行其他任务
2.使用 threading 模块创建和管理线程
2.1创建线程:Thread 类的基本用法
在 Python 中,可以使用 threading 模块中的 Thread 类来创建线程。以下是创建线程的基本用法:
import threadingdef my_function():print("This is a thread.")# 创建线程
my_thread = threading.Thread(target=my_function)
2.2线程的启动和执行:start() 方法
通过调用线程对象的 start() 方法来启动线程,让线程开始执行目标函数中的代码:
my_thread.start()
2.3线程的同步和阻塞:join() 方法
可以使用 join() 方法来等待线程执行完成,实现线程的同步和阻塞:
my_thread.join()
print("Thread has finished.")
2.4线程的名称和标识:name 和 ident 属性
每个线程都有一个名称和一个唯一的标识符。可以通过 name 属性获取线程的名称,通过 ident 属性获取线程的标识符:
print("Thread name:", my_thread.name)
print("Thread identifier:", my_thread.ident)
通过以上方法,可以创建、启动和管理线程,并实现线程间的同步和阻塞操作。线程的名称和标识符可以帮助我们对线程进行标识和跟踪,便于调试和管理多线程程序。
3.线程间的通信与同步
3.1使用 Lock 实现线程同步
在多线程编程中,为了避免多个线程同时访问共享资源导致数据混乱或不一致的问题,可以使用 Lock 对象实现线程同步。下面是一个简单的示例:
import threadingcounter = 0
lock = threading.Lock()def increment_counter():global counterwith lock:counter += 1# 创建多个线程并启动
threads = []
for _ in range(5):t = threading.Thread(target=increment_counter)t.start()threads.append(t)# 等待所有线程执行完成
for t in threads:t.join()print("Final counter value:", counter)
在上面的示例中,通过 Lock 对象确保了 counter 变量的原子性操作,避免了多线程同时修改导致的问题。
3.2使用 Queue 实现线程间通信
另一种常见的方式是使用 Queue 实现线程间的通信。Queue 是线程安全的数据结构,可以在多个线程之间安全地传递数据。以下是一个简单的示例:
import threading
import queuedef producer(q):for i in range(5):q.put(i)def consumer(q):while not q.empty():item = q.get()print("Consumed:", item)# 创建队列
q = queue.Queue()# 创建生产者和消费者线程
producer_thread = threading.Thread(target=producer, args=(q,))
consumer_thread = threading.Thread(target=consumer, args=(q))# 启动线程
producer_thread.start()
consumer_thread.start()# 等待线程执行完成
producer_thread.join()
consumer_thread.join()
在上面的示例中,通过 Queue 实现了生产者和消费者模式,生产者向队列中放入数据,消费者从队列中取出数据进行消费,实现了线程间的通信。
4.线程池的使用
使用 ThreadPoolExecutor 可以方便地管理线程池,控制线程数量并提交任务。以下是 ThreadPoolExecutor 的基本用法示例:
from concurrent.futures import ThreadPoolExecutor
import time# 定义一个任务函数
def task(n):print(f"Task {n} started")time.sleep(2)return f"Task {n} completed"# 创建 ThreadPoolExecutor
with ThreadPoolExecutor(max_workers=3) as executor: # 控制线程池大小为3# 提交任务给线程池future1 = executor.submit(task, 1)future2 = executor.submit(task, 2)future3 = executor.submit(task, 3)# 获取任务执行结果print(future1.result())print(future2.result())print(future3.result())
在上面的示例中,通过 ThreadPoolExecutor 创建了一个最大容纳3个线程的线程池。然后使用 submit() 方法提交了三个任务,并使用 result() 方法获取任务执行结果。
如果想要控制线程池的大小,可以将 max_workers 参数设置为所需的线程数量。通过 ThreadPoolExecutor 可以方便地管理线程池,提高多线程编程的效率。
5.理解全局解释器锁(GIL)对多线程的影响
全局解释器锁(GIL)是在 CPython 解释器中使用的一种机制,它对多线程并发执行产生了一定的限制和影响。以下是关于 GIL 的作用、原理以及对多线程并发执行的限制:
5.1作用和原理:
- 作用: GIL 的作用是确保在解释器级别上,同一时刻只有一个线程可以执行 Python 字节码。这意味着在多核处理器上,Python 程序不能利用多个 CPU 核心同时执行线程。
- 原理: 在 CPython 中,GIL 是由一个互斥锁来实现的。当一个线程获得了 GIL 后,其他线程就无法在同一时间执行 Python 字节码,直到持有 GIL 的线程释放锁。
5.2对多线程并发执行的限制:
- 性能影响: 由于同一时刻只有一个线程可以执行 Python 字节码,因此在多核 CPU 上,并发执行的效率受到限制。特别是对于计算密集型的多线程任务,GIL 可能导致性能瓶颈。
- IO 密集型任务的影响较小: 对于涉及大量 IO 操作的线程,GIL 的影响相对较小,因为在 IO 操作时,线程会主动释放 GIL,让其他线程执行。
- 影响解决方案: 为了充分利用多核 CPU,可以使用多进程、使用其他语言的扩展模块(如使用 C/C++ 编写的扩展模块)或者使用异步编程(如 asyncio)等方式来规避 GIL 的影响。
总之,GIL 的存在使得在 CPython 中的多线程并发执行受到了一定的限制,开发者需要根据具体情况选择合适的解决方案来充分利用多核 CPU 资源。
6.多线程中的常见问题与解决方法
当涉及到多线程编程中的常见问题如死锁(Deadlock)以及线程间数据共享与安全访问时,以下是代码示例和解释:
6.1死锁(Deadlock)的原因及避免方法:
原因: 死锁是指两个或多个线程互相等待对方释放资源而无法继续执行的情况。
避免方法: 避免死锁的一种常见方法是确保线程获取资源的顺序是一致的,或者使用超时机制来打破死锁。下面是一个简单的示例:
import threading# 创建两个锁
lock1 = threading.Lock()
lock2 = threading.Lock()def thread1():lock1.acquire()print("Thread 1 acquired lock 1")# 假设这里需要一段时间lock2.acquire()print("Thread 1 acquired lock 2")lock2.release()lock1.release()def thread2():lock2.acquire()print("Thread 2 acquired lock 2")# 假设这里需要一段时间lock1.acquire()print("Thread 2 acquired lock 1")lock1.release()lock2.release()t1 = threading.Thread(target=thread1)
t2 = threading.Thread(target=thread2)t1.start()
t2.start()t1.join()
t2.join()
在上述示例中,如果 thread1 和 thread2 同时运行,由于它们试图以不同的顺序获取锁,可能导致死锁。为了避免死锁,可以尝试统一锁的获取顺序。
6.2线程间数据共享与安全访问:
在多线程编程中,线程之间共享数据时需要确保线程安全,可以使用互斥锁来保护共享资源。下面是一个简单的示例:
import threadingcounter = 0
lock = threading.Lock()def increment_counter():global counterfor _ in range(100000):lock.acquire()counter += 1lock.release()threads = []
for _ in range(10):t = threading.Thread(target=increment_counter)threads.append(t)for t in threads:t.start()for t in threads:t.join()print("Final counter value:", counter)
在这个示例中,我们使用互斥锁 lock 来确保对 counter 全局变量的安全访问。每个线程在增加 counter 值之前获取锁,操作完成后释放锁,从而避免竞态条件。这样可以确保线程安全地访问共享资源。
7.线程越多越好么?
在Python中使用多线程可以提高程序的并发性,但并不意味着线程越多越好。这是因为 Python 中的全局解释器锁(Global Interpreter Lock,GIL)限制了同一时间只有一个线程可以执行 Python 字节码,因此多线程并不能充分利用多核处理器的优势。
虽然多线程在某些场景下可以提高效率,比如I/O密集型任务,但如果是CPU密集型任务(如大量计算),多线程并不能有效提升性能。此外,线程数过多也会增加线程切换的开销,并可能导致系统资源竞争和调度开销,进而影响整体性能。
因此,在决定使用多线程时,需要考虑以下几点:
- 任务类型:针对不同类型的任务选择合适的并发模型,如I/O密集型任务可以考虑使用多线程,而CPU密集型任务可能更适合使用多进程。
- 平台和环境:要考虑程序运行的平台和环境对多线程的支持情况,以及是否受到 GIL 的影响。
- 系统资源:合理评估系统资源(CPU、内存等)的使用情况,避免过多线程导致资源浪费和性能下降。
综上所述,虽然多线程可以在适当的情况下提高程序的并发性和效率,但并不意味着线程越多越好。在实际应用中,需要根据具体情况慎重考虑线程数量,并结合任务类型、系统资源和性能需求进行合理的设计和调优。
8.面试问题:解析Python中的GIL(全局解释器锁)是什么?它如何影响多线程编程?
问题描述:
请解释 Python 中的 GIL 是什么,以及它如何影响多线程编程。同时,讨论在受 GIL 限制的情况下如何提高 Python 多线程程序的性能。
详细答案:
-
GIL 是什么?
- GIL 是全局解释器锁(Global Interpreter Lock)的缩写,是 Python 解释器中的一个机制。它的作用是保证在解释器级别同一时刻只有一个线程执行 Python 字节码,从而防止多线程同时执行字节码导致的数据竞争和不一致性。
-
GIL 对多线程编程的影响:
- 由于 GIL 的存在,Python 中的多线程无法利用多核处理器来实现真正的并行执行。即使有多个线程,它们依然是以串行的方式执行,因为同一时刻只有一个线程能够获取到 GIL。
-
提高 Python 多线程程序性能的方法:
- 使用多进程替代多线程: Python 中的多进程可以绕过 GIL 的限制,实现真正的并行执行。
- 使用 C 扩展或 Cython: 将性能关键的部分使用 C 语言或 Cython 编写,可以减少对 GIL 的依赖,提高性能。
- 使用异步编程: 使用异步编程库(如 asyncio、aiohttp)可以在不受 GIL 影响的情况下实现并发执行。
9.多线程实战示例
当涉及到使用Python多线程的实际项目示例时,一个常见的场景是同时下载多个文件并将它们保存到本地。在这个示例中,我们将创建一个简单的多线程下载器,每个线程负责下载一个文件,并最后将它们保存到本地。
9.1项目说明:
我们将使用Python的threading模块来实现多线程下载器。每个线程将会下载一个文件,然后将文件保存到指定的目录。为了模拟真实下载过程,我们会使用一个虚拟的文件URL列表。
9.2完整代码示例:
import threading
import requests
import os# 虚拟文件URL列表
file_urls = ["https://www.example.com/file1.txt","https://www.example.com/file2.txt","https://www.example.com/file3.txt"
]# 下载函数
def download_file(url, save_path):response = requests.get(url)with open(save_path, 'wb') as file:file.write(response.content)print(f"Downloaded {url} and saved to {save_path}")# 下载器类
class DownloaderThread(threading.Thread):def __init__(self, url, save_path):threading.Thread.__init__(self)self.url = urlself.save_path = save_pathdef run(self):download_file(self.url, self.save_path)# 创建保存文件的目录
download_dir = "downloads"
os.makedirs(download_dir, exist_ok=True)# 创建并启动多个下载线程
threads = []
for i, url in enumerate(file_urls):file_name = f"file{i+1}.txt"save_path = os.path.join(download_dir, file_name)downloader = DownloaderThread(url, save_path)threads.append(downloader)downloader.start()# 等待所有线程完成下载
for thread in threads:thread.join()print("All downloads completed!")
9.3代码说明:
- 定义了一个
download_file函数用于下载文件,并将其保存到本地。 - 创建了一个
DownloaderThread类,继承自threading.Thread,用于表示下载线程。每个线程负责下载一个文件。 - 创建了一个
download_dir目录用于保存下载的文件。 - 遍历虚拟文件URL列表,为每个文件创建一个下载线程,并启动下载。
- 最后等待所有线程完成下载,并打印提示信息。
通过这个示例,您可以看到如何使用Python多线程实现一个简单的文件下载器,同时了解了如何在实际项目中应用多线程进行并发处理。请注意,对于大规模文件下载或需要更复杂逻辑的情况,可能需要进一步优化和改进。
相关文章:
Python教程:一文掌握Python多线程(很详细)
目录 1.什么是多线程? 1.1多线程与单线程的区别 1.2 Python 中的多线程实现方式 2.使用 threading 模块创建和管理线程 2.1创建线程:Thread 类的基本用法 2.2线程的启动和执行:start() 方法 2.3线程的同步和阻塞:join() 方…...
华为防火墙配置指引超详细(包含安全配置部分)以USG6320为例
华为防火墙USG6320 华为防火墙USG6320是一款高性能、高可靠的下一代防火墙,适用于中小型企业、分支机构等场景。该防火墙支持多种安全功能,可以有效抵御网络攻击,保护网络安全。 目录 华为防火墙USG6320 1. 初始配置 2. 安全策略配置 3. 防火墙功能配置 4. 高可用性配…...
(含react-draggable库以及相关BUG如何解决)固定在左上方某盒子内(如按钮)添加可拖动功能,使用react hook语法实现
原生写法 // 封装组件 import React, { useState, useRef } from react;const DraggableModal ({ children }) > {const [position, setPosition] useState({ x: 0, y: 0 });const modalRef useRef(null);const handleMouseDown (e) > {const modal modalRef.curre…...
选择最佳图像处理工具OpenCV、JAI、ImageJ、Thumbnailator和Graphics2D
文章目录 1、前言2、 图像处理工具效果对比2.1 Graphics2D实现2.2 Thumbnailator实现2.3 ImageJ实现2.4 JAI(Java Advanced Imaging)实现2.5 OpenCV实现 3、图像处理工具结果 1、前言 SVD(stable video diffusion)开放了图生视频的API,但是限…...
微信小程序版本更新检测
app.vue文件 <script>export default {onLaunch: function() {console.log(App Launch)// #ifdef MP-WEIXINthis.getUpdateManager();// #endif},methods: {// 检测小程序更新getUpdateManager() {const updateManager wx.getUpdateManager();updateManager.onCheckFor…...
【每日力扣】343. 整数拆分与63. 不同路径 II
🔥 个人主页: 黑洞晓威 😀你不必等到非常厉害,才敢开始,你需要开始,才会变的非常厉害 343. 整数拆分 给定一个正整数 n ,将其拆分为 k 个 正整数 的和( k > 2 ),并使…...
洛谷 Cut Ribbon
思路:我们可以看出,这是一道完全背包问题,但是呢,有一点需要注意:那就是我们在装背包的时候并不能保证一定能装满背包,但是这里的背包要求是让我们装满的,所以我们需要判断这个背包装满才行&…...
#AS,idea,maven,gradle
Jdk,sdk。提前都是需要下好的。 Maven与gradle的思考: 用AS开发app时,gradle本就有,自己也可以指定,AGP同样。要注意gradle,AGP,jdk版本的事情。还有依赖库。 用idea开发网络程序时,也有内置的maven&…...
FPGA结构与片上资源
文章目录 0.总览1.可配置逻辑块CLB1.1 6输入查找表(LUT6)1.2 选择器(MUX)1.3 进位链(Carry Chain)1.4 触发器(Flip-Flop) 2.可编程I/O单元2.1 I/O物理级2.2 I/O逻辑级 3.布线资源4.其…...
【分布式】——分布式事务
分布式事务 ⭐⭐⭐⭐⭐⭐ Github主页👉https://github.com/A-BigTree 笔记链接👉https://github.com/A-BigTree/tree-learning-notes ⭐⭐⭐⭐⭐⭐ Spring专栏👉https://blog.csdn.net/weixin_53580595/category_12279588.html SpringMVC专…...
第6章:“让我们思考这个”的提示
“让我们思考这个”这一提示词,是深度对话的钥匙,鼓励ChatGPT生成反思性、沉思性的文本。 对于论文写作、诗歌创作或创意任务的完成,非常实用。 当你想要深究某主题时,只需向ChatGPT提问。 它会基于提示,结合算法和…...
安卓Activity上滑关闭效果实现
最近在做一个屏保功能,需要支持如图的上滑关闭功能。 因为屏保是可以左右滑动切换的,内部是一个viewpager 做这个效果的时候,关键就是要注意外层拦截触摸事件时,需要有条件的拦截,不能影响到内部viewpager的滑动处理…...
使用conda管理python环境
为什么需要管理环境? 每个python程序依赖的库版本可能不同,因此我们需要隔离不同的环境。 创建环境: conda create --name myenv python3.8这将创建一个名为myenv的新环境,并在其中安装Python 3.8版本。 列出所有环境…...
MR混合现实情景实训教学系统在军事演练课堂中的教学应用
MR混合现实情景实训教学系统在军事演练课堂中的教学应用具有以下优势: 1. 增强现实感:通过MR技术,学生可以在军事演练中更真实地感受到战场环境,增强他们的实战经验。 2. 提高训练效率:通过MR技术,可以模…...
vant checkbox 复选框 样式改写
修改前 修改后 基于 vant: 4.8.3 unocss: 0.53.4 <van-checkbox-group v-model"query.zczb" shape"square" class"text-16 w-100% flex flex-wrap"><template v-for"item in registerCapitalOption"><v…...
物联网实战--入门篇之(一)物联网概述
目录 一、前言 二、知识梳理 三、项目体验 四、项目分解 一、前言 近几年很多学校开设了物联网专业,但是确却地讲,物联网属于一个领域,包含了很多的专业或者说技能树,例如计算机、电子设计、传感器、单片机、网…...
将yolov5s部署到安卓上实战经验总结
最近需要在手机端实现一个目标检测的功能,于是选择了小巧又在目标检测方面表现很好的yolov5s,官网下载yolov5代码,用自己做的数据集进行了训练,然后把模型转换成torchscript格式,这些过程网上都有很多讲解,…...
算法日记————对顶堆(4道题)
对顶堆的作用主要在于动态维护第k大的数字,考虑使用两个优先队列,一个大9999999999根堆一个小根堆,小根堆维护大于等于第k大的数字的数,它的堆顶就是堆内最小,第k大的数字,另外一个大根堆维护小于等于k的数…...
【I.MX6ULL移植】Ubuntu-base根文件系统移植
1.下载Ubuntu16.04根文件系统 http://cdimage.ubuntu.com/ 1 2 3 4 5 2.解压ubuntu base 根文件系统 为了存放 ubuntu base 根文件系统,先在 PC 的 Ubuntu 系统中的 nfs 目录下创建一个名为 ubuntu_rootfs 的目录,命令如下: 【注意&…...
unity3d for web
时光噶然 一晃好多年过去了(干了5年的u3d游戏),记得最后一次使用的版本好像是 unity 2017。 那个是 unity3d for webgl 还需要装个插件。用起来很蛋疼。 最近做一个小项目 在选择是用 Layabox 还是 cocosCreate 的时候 我想起了老战友 Uni…...
Python爬虫实战:研究MechanicalSoup库相关技术
一、MechanicalSoup 库概述 1.1 库简介 MechanicalSoup 是一个 Python 库,专为自动化交互网站而设计。它结合了 requests 的 HTTP 请求能力和 BeautifulSoup 的 HTML 解析能力,提供了直观的 API,让我们可以像人类用户一样浏览网页、填写表单和提交请求。 1.2 主要功能特点…...
转转集团旗下首家二手多品类循环仓店“超级转转”开业
6月9日,国内领先的循环经济企业转转集团旗下首家二手多品类循环仓店“超级转转”正式开业。 转转集团创始人兼CEO黄炜、转转循环时尚发起人朱珠、转转集团COO兼红布林CEO胡伟琨、王府井集团副总裁祝捷等出席了开业剪彩仪式。 据「TMT星球」了解,“超级…...
postgresql|数据库|只读用户的创建和删除(备忘)
CREATE USER read_only WITH PASSWORD 密码 -- 连接到xxx数据库 \c xxx -- 授予对xxx数据库的只读权限 GRANT CONNECT ON DATABASE xxx TO read_only; GRANT USAGE ON SCHEMA public TO read_only; GRANT SELECT ON ALL TABLES IN SCHEMA public TO read_only; GRANT EXECUTE O…...
【决胜公务员考试】求职OMG——见面课测验1
2025最新版!!!6.8截至答题,大家注意呀! 博主码字不易点个关注吧,祝期末顺利~~ 1.单选题(2分) 下列说法错误的是:( B ) A.选调生属于公务员系统 B.公务员属于事业编 C.选调生有基层锻炼的要求 D…...
Web 架构之 CDN 加速原理与落地实践
文章目录 一、思维导图二、正文内容(一)CDN 基础概念1. 定义2. 组成部分 (二)CDN 加速原理1. 请求路由2. 内容缓存3. 内容更新 (三)CDN 落地实践1. 选择 CDN 服务商2. 配置 CDN3. 集成到 Web 架构 …...
Springboot社区养老保险系统小程序
一、前言 随着我国经济迅速发展,人们对手机的需求越来越大,各种手机软件也都在被广泛应用,但是对于手机进行数据信息管理,对于手机的各种软件也是备受用户的喜爱,社区养老保险系统小程序被用户普遍使用,为方…...
JVM虚拟机:内存结构、垃圾回收、性能优化
1、JVM虚拟机的简介 Java 虚拟机(Java Virtual Machine 简称:JVM)是运行所有 Java 程序的抽象计算机,是 Java 语言的运行环境,实现了 Java 程序的跨平台特性。JVM 屏蔽了与具体操作系统平台相关的信息,使得 Java 程序只需生成在 JVM 上运行的目标代码(字节码),就可以…...
【笔记】WSL 中 Rust 安装与测试完整记录
#工作记录 WSL 中 Rust 安装与测试完整记录 1. 运行环境 系统:Ubuntu 24.04 LTS (WSL2)架构:x86_64 (GNU/Linux)Rust 版本:rustc 1.87.0 (2025-05-09)Cargo 版本:cargo 1.87.0 (2025-05-06) 2. 安装 Rust 2.1 使用 Rust 官方安…...
虚拟电厂发展三大趋势:市场化、技术主导、车网互联
市场化:从政策驱动到多元盈利 政策全面赋能 2025年4月,国家发改委、能源局发布《关于加快推进虚拟电厂发展的指导意见》,首次明确虚拟电厂为“独立市场主体”,提出硬性目标:2027年全国调节能力≥2000万千瓦࿰…...
如何更改默认 Crontab 编辑器 ?
在 Linux 领域中,crontab 是您可能经常遇到的一个术语。这个实用程序在类 unix 操作系统上可用,用于调度在预定义时间和间隔自动执行的任务。这对管理员和高级用户非常有益,允许他们自动执行各种系统任务。 编辑 Crontab 文件通常使用文本编…...
