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 threading

def 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 threading

def 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 threading

counter = 0
lock = threading.Lock()

def increment_counter():
    global counter
    with 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 queue

def 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对多线程并发执行的限制:

  1. 性能影响: 由于同一时刻只有一个线程可以执行 Python 字节码,因此在多核 CPU 上,并发执行的效率受到限制。特别是对于计算密集型的多线程任务,GIL 可能导致性能瓶颈。
  2. IO 密集型任务的影响较小: 对于涉及大量 IO 操作的线程,GIL 的影响相对较小,因为在 IO 操作时,线程会主动释放 GIL,让其他线程执行。
  3. 影响解决方案: 为了充分利用多核 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()

在上述示例中,如果 thread1thread2 同时运行,由于它们试图以不同的顺序获取锁,可能导致死锁。为了避免死锁,可以尝试统一锁的获取顺序。

6.2线程间数据共享与安全访问:

在多线程编程中,线程之间共享数据时需要确保线程安全,可以使用互斥锁来保护共享资源。下面是一个简单的示例:

import threading

counter = 0
lock = threading.Lock()

def increment_counter():
    global counter
    for _ in range(100000):
        lock.acquire()
        counter += 1
        lock.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密集型任务(如大量计算),多线程并不能有效提升性能。此外,线程数过多也会增加线程切换的开销,并可能导致系统资源竞争和调度开销,进而影响整体性能。

因此,在决定使用多线程时,需要考虑以下几点:

  1. 任务类型:针对不同类型的任务选择合适的并发模型,如I/O密集型任务可以考虑使用多线程,而CPU密集型任务可能更适合使用多进程。
  2. 平台和环境:要考虑程序运行的平台和环境对多线程的支持情况,以及是否受到 GIL 的影响。
  3. 系统资源:合理评估系统资源(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 = url
        self.save_path = save_path

    def 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代码说明:

  1. 定义了一个download_file函数用于下载文件,并将其保存到本地。
  2. 创建了一个DownloaderThread类,继承自threading.Thread,用于表示下载线程。每个线程负责下载一个文件。
  3. 创建了一个download_dir目录用于保存下载的文件。
  4. 遍历虚拟文件URL列表,为每个文件创建一个下载线程,并启动下载。
  5. 最后等待所有线程完成下载,并打印提示信息。

通过这个示例,您可以看到如何使用Python多线程实现一个简单的文件下载器,同时了解了如何在实际项目中应用多线程进行并发处理。请注意,对于大规模文件下载或需要更复杂逻辑的情况,可能需要进一步优化和改进。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/498904.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

国外的Java面试题和国内的相比谁更卷

前言 有很多朋友很好奇国外的Java面试题长啥样,今天我们就去找5道国外的面试来和国内的对比一下看看谁难一些! 面试题分享 1. Is Java Platform Independent if then how?( Java平台是独立的吗?) Yes, Java is a…

三思多功能智慧综合杆助推上海杨浦区数智化升级

旧貌换新颜。上海三思为上海杨浦区政和路、政悦路、殷高东路等城市道路建成多功能综合智慧杆200余杆,让城市面貌焕然一新,智慧杨浦再上新台阶。 本项目通过集约化建设手段,有力地推动管理部门从粗放式管理向精细化管理转型。项目的实施促进道…

机器学习算法的另一个分支-贝叶斯算法原理(贝叶斯要解决什么问题)

目录 一、贝叶斯简介 二、贝叶斯要解决的问题 三、例子(公式推导) 四、实例 1. 拼写纠正实例 2. 垃圾邮件过滤实例 一、贝叶斯简介 1. 贝叶斯:英国数学家。1702年出生于伦敦,做过神甫。贝叶斯在数学方面主要研究概率论.对于…

Request对象

目录 1、GET方法 2、POST方法 引出问题:我们前面在赋值的时候,都是在一个页面进行赋值,那么怎么样将web1的数据传送到web2中呢,这时候,就要用到request方法了。 作用:Request对象主要是让服务器取得客户…

蓝桥杯算法题练习

1、20世纪有多少个星期一 (1901、1、1——2000、12、31) 方法一:python代码 方法二:excel工具(设置单元格格式,把日期换成周几的形式) 2、100个数相乘,结果有几个0 3、切面条 找规律:对折次数n 弯2^n-1 面…

公众号 服务号 代码创建菜单方法 公众号跳转小程序功能 40027错误的解决

需求 通过代码实现微信公众号的自定义菜单,新增、同步菜单。 分析 其实对一个公众号而言,菜单只要创建一次就可以了,如果你不是服务商,可以直接使用微信提供的代码调试工具,直接发送json字符串,为自己的…

系统开发实训小组作业week5 —— 用例描述与分析

目录 4.3 UC003电影浏览与查询 4.3.1 用例描述 4.3.2 活动图 4.3.3 界面元素 4.3.4 功能 4.4 UC004在线订票 4.4.1 用例描述 4.4.2 活动图 4.4.3 界面元素 4.4.4 功能 4.3 UC003电影浏览与查询 4.3.1 用例描述 用例号 UC003-01 用例名称 电影浏览与查询 用例描述…

App 测试必备 - 建议所有测试人收藏

移动端App性能测试需要关注多个方面,包括响应时间、稳定性、内存使用、CPU使用率、网络性能、电池消耗以及设备兼容性等。通过综合考虑这些方面,并在不同条件下进行全面的测试,可以确保应用程序在各种情况下都能够提供优质的用户体验&#xf…

AI人像写真解决方案,满足企业多样化视觉需求

美摄科技,作为一家专注于人工智能技术研发与应用的高新企业,深知企业对于高质量、高效率视觉内容的需求,特推出AI人像写真解决方案,为企业打开全新的视觉营销通道。 我们的AI人像写真解决方案,基于深度学习和计算机视…

解密EMC与EMI:电磁兼容性与电磁干扰?|深圳比创达电子

在现代电子设备愈发普及的时代,EMC(Electromagnetic Compatibility)和EMI(Electromagnetic Interference)成为了不容忽视的重要议题。本文将从根本概念出发,逐步深入探讨这两个关键领域,带领各位…

【信贷后台管理之登录(一)】

提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 目录结构一、项目搭建二、登录页面1.引入Element-ui2.LoginView.vue组件3.router文件的index.js引入组件 gitee仓库地址 目录结构 一、项目搭建 找到存放项目的文件…

解析旅游者心声:用PySpark和SnowNLP揭秘景区评论的情感秘密

简介: 在本篇博客中,我们将探讨如何利用PySpark和SnowNLP这两个强大的工具来分析大规模的旅游评论数据。通过结合携程和去哪儿的数据作为示例,我们将探索如何从海量的评论中提取有价值的情感信息和洞察。PySpark作为一种分布式计算框架,能够处理大规模的数据集,为我们提供…

Java服务运行在Linux----维护常用命令

想起来哪些再添加上去 查看Java程序进程 jps -l 查出进程后根据pid 查询程序所在目录 pwdx 31313 根据端口查找PID 根据pid杀死程序 kill -p 31313 查看目录下所有包含9527的文件 grep -rn 9527 查看磁盘空间 查找文件名"nginx"文件或模糊查找"*nginx*&quo…

Mysql中如何显示第几周

在数据分析中,经常需要对日期和时间进行格式化处理,以便更直观地展示和理解数据。 MySQL 5.7提供了强大的DATE_FORMAT函数,允许用户根据特定的格式字符串来显示日期和时间。 Week函数 最直接的是使用YEAR、WEEK函数 SELECTYEAR(NOW()) AS C…

健身运动耳机哪个牌子好?力荐五大品质翘楚的精品

健身已经成为许多人追求健康与活力的重要方式,而在健身的过程中,一款优质的耳机不仅能让你沉浸于音乐的世界,更能提升运动体验,激发无限潜能,那么如何选择一款既适合运动又品质卓越的耳机呢?今天我这个健身…

【热门话题】Yarn:新一代JavaScript包管理器的安装与使用

🌈个人主页: 鑫宝Code 🔥热门专栏: 闲话杂谈| 炫酷HTML | JavaScript基础 ​💫个人格言: "如无必要,勿增实体" 文章目录 Yarn:新一代JavaScript包管理器的安装与使用引言一、Yarn的安装1. 系…

element-ui inputNumber 组件源码分享

今日简单分享 inputNumber 组件的实现原理,主要从以下四个方面来分享: 1、inputNumber 组件的页面结构 2、inputNumber 组件的属性 3、inputNumber 组件的事件 4、inputNumber 组件的方法 一、inputNumber 组件的页面结构。 二、inputNumber 组件的…

linux安装Zookeeper的详细步骤

1.Java环境确认 确保已经安装了Java环境,没有的自行安装 2.官网下载包 Apache ZooKeeper 3.安装 3.1上传到linux,解压 我的目录为/root/apache-zookeeper-3.8.4-bin 进入到/root/apache-zookeeper-3.8.4-bin/conf目录下,执行命令复制zoo…

由浅到深认识Java语言(44):Junit单元测试

该文章Github地址:https://github.com/AntonyCheng/java-notes 在此介绍一下作者开源的SpringBoot项目初始化模板(Github仓库地址:https://github.com/AntonyCheng/spring-boot-init-template & CSDN文章地址:https://blog.c…

硬件12、PCB模块化布局

模块画布局其实就是根据原理图中绘制的小模块,比如3.3V芯片及其外围电路部分的元器件在PCB中放在一起进行布线,会方便很多 1、最好打开分屏,一边在原理图中选中模块,一边在PCB中绘制 2、选中原理图中的模块的所有元件&#xff0…
最新文章