python基础篇:多线程的基本使用

本文涉及的产品
Serverless 应用引擎 SAE,800核*时 1600GiB*时
性能测试 PTS,5000VUM额度
云原生网关 MSE Higress,422元/月
简介: python基础篇:多线程的基本使用

Python多线程是一种并发编程的方式,可以让程序同时执行多个任务。在Python中,多线程可以使用标准库中的threading模块来实现。本文将介绍如何使用threading模块来创建和管理线程。

创建线程

在Python中,创建线程可以通过创建Thread对象来实现。Thread对象有一个target参数,指定线程要执行的函数。例如:

import threading

def print_numbers():
    for i in range(10):
        print(i)

thread = threading.Thread(target=print_numbers)
thread.start()

在这个例子中,我们创建了一个名为print_numbers的函数,并将其作为Thread对象的target参数传递。然后我们调用start()方法来启动线程。当线程启动后,它将调用print_numbers函数来执行任务。

传递参数

有时候我们需要在线程中传递参数。可以通过在Thread对象的args参数中传递参数。例如:

import threading

def print_numbers(start, end):
    for i in range(start, end):
        print(i)

thread = threading.Thread(target=print_numbers, args=(1, 10))
thread.start()

在这个例子中,我们传递了start和end两个参数给print_numbers函数。将这些参数作为元组传递给args参数,即args=(1, 10)。在print_numbers函数中,我们使用这些参数来打印数字。

线程同步

在多线程编程中,线程之间可能会访问共享资源。如果没有适当的同步机制,可能会导致竞态条件和死锁等问题。Python提供了多种同步机制,例如锁(Lock)、信号量(Semaphore)和条件变量(Condition)等。这里以锁为例,介绍如何在Python中使用锁来实现线程同步。

import threading

x = 0
lock = threading.Lock()

def increment():
    global x
    for i in range(100000):
        lock.acquire()
        x += 1
        lock.release()

def decrement():
    global x
    for i in range(100000):
        lock.acquire()
        x -= 1
        lock.release()

thread1 = threading.Thread(target=increment)
thread2 = threading.Thread(target=decrement)
thread1.start()
thread2.start()
thread1.join()
thread2.join()
print(x)

在这个例子中,我们定义了两个函数increment和decrement,分别对全局变量x进行加一和减一操作。为了避免竞态条件,我们使用Lock对象来控制对x的访问。在increment和decrement函数中,我们调用acquire()方法来获取锁,然后进行相应的操作,最后调用release()方法来释放锁。这样,每个线程在访问x时都会先获得锁,从而避免了竞态条件。
在这个例子中,我们创建了两个线程thread1和thread2,分别执行increment和decrement函数。我们通过调用start()方法来启动线程,然后通过调用join()方法来等待线程执行完毕。最后,我们打印x的值,以检查线程同步是否正确。

线程池

在Python中,可以使用ThreadPoolExecutor类来创建线程池,以便同时执行多个任务。ThreadPoolExecutor可以自动管理线程的数量和生命周期,从而避免创建过多线程导致系统负载过高的问题。以下是一个简单的示例:

import concurrent.futures

def print_numbers(start, end):
    for i in range(start, end):
        print(i)

with concurrent.futures.ThreadPoolExecutor(max_workers=2) as executor:
    executor.submit(print_numbers, 1, 5)
    executor.submit(print_numbers, 6, 10)

在这个例子中,我们使用ThreadPoolExecutor创建了一个最大工作线程数为2的线程池。然后我们通过调用submit()方法来提交两个任务,分别打印1到5和6到10之间的数字。submit()方法返回一个Future对象,用于表示任务的执行状态和结果。在这个例子中,我们没有使用Future对象来获取任务的结果,而是直接打印数字。

Future对象的作用

Future对象是在Python标准库中的concurrent.futures模块中提供的一种异步编程工具,用于表示一个尚未完成的异步操作的状态和结果。

当一个任务被提交到线程池或者进程池中时,会立即返回一个Future对象。这个对象可以用于查询任务的状态和结果。Future对象有以下几种状态:

  • PENDING:任务尚未开始执行。
  • RUNNING:任务正在执行。
  • CANCELLED:任务已被取消。
  • FINISHED:任务已完成,可能成功也可能失败。

当任务完成后,Future对象会保存任务的结果,可以通过调用result()方法来获取。如果任务尚未完成,调用result()方法会阻塞当前线程,直到任务完成并返回结果或者抛出异常。

除了result()方法之外,Future对象还提供了一些其他的方法,例如cancel()方法可以用于取消任务,done()方法用于检查任务是否完成,add_done_callback()方法可以用于注册回调函数,当任务完成时自动调用。

综合演示例子

面是一个综合的例子,演示如何使用concurrent.futures模块中的ThreadPoolExecutor和Future对象来实现并发下载图片的功能。

import requests
import concurrent.futures

def download_image(url):
    response = requests.get(url)
    if response.status_code == 200:
        filename = url.split("/")[-1]
        with open(filename, "wb") as f:
            f.write(response.content)
        return filename

urls = [
    "https://picsum.photos/200/300",
    "https://picsum.photos/250/350",
    "https://picsum.photos/300/400",
    "https://picsum.photos/350/450",
    "https://picsum.photos/400/500",
]

with concurrent.futures.ThreadPoolExecutor() as executor:
    futures = [executor.submit(download_image, url) for url in urls]
    for future in concurrent.futures.as_completed(futures):
        filename = future.result()
        if filename:
            print(f"{filename} downloaded successfully!")
  • 在这个例子中,我们定义了一个download_image()函数,用于下载指定URL对应的图片,并将图片保存到本地文件系统中。然后我们定义了一个URL列表urls,包含了要下载的五张图片的URL地址。

  • 接下来,我们使用ThreadPoolExecutor创建一个线程池,并使用submit()方法提交五个下载任务。submit()方法返回一个Future对象,代表了提交的任务。我们将所有的Future对象保存在一个列表中。

  • 然后,我们使用as_completed()函数遍历所有的Future对象,并在每个Future对象完成后调用result()方法获取结果。如果下载成功,就打印出文件名。

通过使用ThreadPoolExecutor和Future对象,我们可以同时下载多个图片,从而提高下载速度。同时,使用Future对象可以让我们方便地处理每个任务的结果,从而实现更加灵活的异步编程。

总结

本文介绍了如何在Python中使用threading模块来创建和管理线程,包括传递参数、线程同步和线程池等方面。同时,我们也介绍了一些常见的多线程编程问题,例如竞态条件和死锁等,以及如何使用锁来避免这些问题。在实际编程中,需要根据具体情况选择适当的同步机制和线程池配置,以提高程序的性能和稳定性。

目录
相关文章
|
6天前
|
安全 Python
告别低效编程!Python线程与进程并发技术详解,让你的代码飞起来!
【7月更文挑战第9天】Python并发编程提升效率:**理解并发与并行,线程借助`threading`模块处理IO密集型任务,受限于GIL;进程用`multiprocessing`实现并行,绕过GIL限制。示例展示线程和进程创建及同步。选择合适模型,注意线程安全,利用多核,优化性能,实现高效并发编程。
20 3
|
6天前
|
安全 数据安全/隐私保护 数据中心
Python并发编程大挑战:线程安全VS进程隔离,你的选择影响深远!
【7月更文挑战第9天】Python并发:线程共享内存,高效但需处理线程安全(GIL限制并发),适合IO密集型;进程独立内存,安全但通信复杂,适合CPU密集型。使用`threading.Lock`保证线程安全,`multiprocessing.Queue`实现进程间通信。选择取决于任务性质和性能需求。
19 1
|
6天前
|
Python
解锁Python并发新世界:线程与进程的并行艺术,让你的应用性能翻倍!
【7月更文挑战第9天】并发编程**是同时执行多个任务的技术,提升程序效率。Python的**threading**模块支持多线程,适合IO密集型任务,但受GIL限制。**multiprocessing**模块允许多进程并行,绕过GIL,适用于CPU密集型任务。例如,计算平方和,多线程版本使用`threading`分割工作并同步结果;多进程版本利用`multiprocessing.Pool`分块计算再合并。正确选择能优化应用性能。
|
7天前
|
安全 Java 调度
「Python入门」Python多线程
1. **线程与进程区别**:线程共享内存,进程独立;线程启动快,多线程效率高于多进程。 2. **多线程使用**:直接使用Thread类,通过`target`指定函数,`args`传递参数;或继承Thread,重写`run`方法。 3. **守护线程**:设置`setDaemon(True)`,主线程结束时,守护线程一同结束。 4. **join线程同步**:主线程等待子线程完成,如`t.join()`。 5. **线程锁**(Mutex):防止数据竞争,确保同一时间只有一个线程访问共享资源。 6. **RLock(递归锁)**:允许多次锁定,用于需要多次加锁的递归操作。
16 1
「Python入门」Python多线程
|
1天前
|
消息中间件 安全 数据处理
Python中的并发编程:理解多线程与多进程的区别与应用
在Python编程中,理解并发编程是提高程序性能和响应速度的关键。本文将深入探讨多线程和多进程的区别、适用场景及实际应用,帮助开发者更好地利用Python进行并发编程。
|
2天前
|
缓存 并行计算 监控
了解 Python 线程
【7月更文挑战第8天】在Python多线程编程中,`threading`模块允许我们获取当前线程名字,通过`current_thread().name`获取。线程名字有助于调试、日志和资源管理。示例代码展示了如何创建线程并打印其名字。在实际应用中,线程命名应清晰、唯一且避免特殊字符,以提高代码可读性和维护性。多线程编程需注意线程安全、死锁、性能优化等问题。通过合理设计和测试,可以利用多线程提高程序并发性和效率。
6 1
|
7天前
|
数据处理 调度 Python
Python并发编程实战指南:深入理解线程(threading)与进程(multiprocessing)的奥秘,打造高效并发应用!
【7月更文挑战第8天】Python并发编程探索:使用`threading`模块创建线程处理任务,虽受限于GIL,适合I/O密集型工作。而`multiprocessing`模块通过进程实现多核利用,适用于CPU密集型任务。通过实例展示了线程和进程的创建与同步,强调了根据任务类型选择合适并发模型的重要性。
|
5天前
|
数据库 数据安全/隐私保护 C++
Python并发编程实战:线程(threading)VS进程(multiprocessing),谁才是并发之王?
【7月更文挑战第10天】Python并发对比:线程轻量级,适合I/O密集型任务,但受GIL限制;进程绕过GIL,擅CPU密集型,但通信成本高。选择取决于应用场景,线程利于数据共享,进程利于多核利用。并发无“王者”,灵活运用方为上策。
|
6天前
|
安全 API 调度
深度剖析:Python并发编程中的线程与进程,那些你不可不知的使用技巧与限制!
【7月更文挑战第9天】Python并发:线程适合IO密集型任务,利用GIL下的多线程同步,如示例中使用锁。进程适用于CPU密集型,通过multiprocessing模块实现多进程,利用进程间通信如队列。线程受限于GIL,进程间通信成本高。选择取决于任务需求和性能目标。
12 2
|
7天前
|
大数据 API 数据处理
Python高手都在用的并发秘籍:解锁线程与进程的终极奥义,性能飙升不是梦!
【7月更文挑战第8天】Python并发编程提升性能,线程(threading)适合I/O密集型任务,如网络请求,通过`start()`和`join()`实现并发。进程(multiprocessing)利用多核CPU,适用于CPU密集型任务,如大数据处理。结合两者可优化混合任务,实现最佳并发效果。
10 1

热门文章

最新文章

  • 1
    @SneakyThrows 是 Lombok 库中的一个注解
    5
  • 2
    在会议系统工程中,Python可以用于多种任务,如网络请求(用于视频会议的连接和会议数据的传输)、数据分析(用于分析会议参与者的行为或会议效果)等。
    9
  • 3
    在可视会议系统工程中,系统工程方法可以帮助我们系统地规划、设计和实现一个高效、可靠的可视会议系统。
    10
  • 4
    我们可以从系统工程的角度来讨论如何优化组织架构,并给出一些可能涉及的Python应用领域的示例。
    7
  • 5
    在环境治理领域,污染治理系统工程旨在通过系统的方法来解决环境污染问题。这通常包括污染源的识别、污染物的监测、治理技术的选择、治理效果的评估等多个环节。
    13
  • 6
    我将提供一个简化的Python代码示例和详解,以展示如何使用Python和Django框架来构建智能化小区综合物业管理系统的一部分功能。
    8
  • 7
    在系统工程中,软件测试是一个至关重要的环节,它确保软件的质量、可靠性和性能。软件测试通常包括多个阶段,如单元测试、集成测试、系统测试和验收测试等。
    14
  • 8
    在软件部署阶段,系统工程的目标是确保软件能够顺利、稳定地部署到目标环境中,并满足用户的需求。
    11
  • 9
    航空航天领域,系统工程被用于设计复杂的飞行器和系统。这包括飞行器的结构、推进系统、控制系统等。
    12
  • 10
    在通讯系统工程中,这通常包括硬件、软件、网络协议、数据传输等多个方面的设计和实现。
    9