您当前的位置: 首页 > 服务

Python多线程

博客园 来源:2023-08-26 11:25:11

当进行多线程编程时,涉及到以下几个关键概念和操作


(相关资料图)

多线程是指在一个进程中同时运行多个线程,每个线程都可以执行不同的任务。线程是操作系统能够进行运算调度的最小单位,它是进程中的一个实体,可以被操作系统独立调度。

以下是与多线程相关的一些核心概念:

1. **进程和线程**: - 进程(Process)是一个执行中的程序,它拥有独立的内存空间和系统资源。一个进程可以包含多个线程。 - 线程(Thread)是一个轻量级的执行单元,线程在进程内共享内存空间,可以更快地切换和执行。

2. **并发和并行**: - 并发(Concurrency)是指多个任务交替执行的状态,但不同任务的执行时间可能重叠,侧重于提高资源利用率。 - 并行(Parallelism)是指多个任务同时执行的状态,侧重于提高执行速度。

3. **GIL(Global Interpreter Lock)**: - 在 CPython(标准的 Python 解释器)中,由于 GIL 的存在,一次只允许一个线程执行 Python 代码。这意味着在多线程环境中,同一时刻只有一个线程能够执行 Python 代码,不同于多核 CPU 上的真正并行。 - GIL 使得 Python 中的多线程在一些 CPU 密集型任务上效率不高,但对于 I/O 密集型任务仍然有优势。

4. **线程同步和竞争条件**: - 多线程环境下,不同线程可能同时访问共享的资源,可能导致竞争条件(Race Condition),从而导致数据不一致和错误。 - 线程同步机制(如锁、信号量、条件变量等)用于控制多个线程对共享资源的访问,以避免竞争条件。

5. **线程池**: - 线程池是一种管理和复用线程的技术,它在程序启动时创建一组线程,并将任务分配给这些线程,避免了频繁创建和销毁线程的开销。

多线程适用于以下情况:

- 处理多任务,同时执行多个相对独立的任务。- 提高程序的响应能力,使用户界面保持活跃,同时后台执行任务。- 在多核 CPU 上,实现一定程度的并行计算。- 处理 I/O 密集型任务,如网络通信、文件读写,以充分利用等待的时间。

需要注意的是,多线程编程需要小心处理线程间的竞争条件和同步问题,以避免出现难以调试的问题。在 Python 中,使用 `threading` 模块可以进行多线程编程:

1. **创建多线程**

使用 `threading.Thread` 类可以创建新的线程。通过传递一个函数(或可调用对象)作为参数给线程类,这个函数将成为线程的执行体。

import threadingdef worker():    print("Worker thread")thread = threading.Thread(target=worker)

2. **主线程**:

主线程是程序启动时默认创建的线程。它是第一个开始执行的线程,通常用于初始化、调度和管理其他线程。

3. **阻塞线程**:

在多线程环境中,有时需要等待其他线程完成某些操作,或者等待一段时间后再继续执行。可以使用 `thread.join()` 方法来阻塞当前线程,直到指定的线程执行完毕。

thread.join()  # 阻塞当前线程,直到 thread 执行完毕

4. **判断线程是否活动**:

使用 `threading.Thread.is_alive()` 方法可以判断一个线程是否处于活动状态(正在执行)。

if thread.is_alive():    print("Thread is still active")

5. **线程同步**:

在多线程环境中,由于多个线程可能共享相同的资源,可能会导致竞争条件和数据不一致问题。为了避免这些问题,可以使用锁(`threading.Lock`)等同步机制来保护共享资源。

lock = threading.Lock()def worker():    with lock:        # 保护共享资源的代码块

创建多个线程的两种方法:

1. **创建线程实例**

通过创建多个 `threading.Thread` 类的实例来创建多个线程。

thread1 = threading.Thread(target=worker1)thread2 = threading.Thread(target=worker2)

使用 threading.Thread类创建多线程时,可以传递一些参数来控制线程的行为和执行。以下是 threading.Thread构造函数中一些重要参数的含义:

    • 这是一个必需的参数,用于指定线程要执行的函数或可调用对象。线程将会运行这个函数。
    • 示例:threading.Thread(target=my_function)

下面是一个示例,演示了如何创建一个带参数的多线程:

import threadingdef print_numbers(start, end):    for i in range(start, end):        print(i)thread = threading.Thread(target=print_numbers, args=(1, 6))thread.start()thread.join()

2. **继承 Thread 类**:

创建一个继承自 `threading.Thread` 类的子类,重写 `run()` 方法,然后通过实例化子类来创建线程。

class MyThread(threading.Thread):    def run(self):        # 线程执行的内容thread1 = MyThread()thread2 = MyThread()

综合以上概念,以下是一个完整的示例,演示了如何创建多个线程、主线程的行为、阻塞线程、判断线程是否活动以及使用锁进行线程同步:

import threadingimport timedef worker():    print(f"Worker thread started by {threading.current_thread().name}")    time.sleep(2)    print(f"Worker thread finished by {threading.current_thread().name}")lock = threading.Lock()def synchronized_worker():    with lock:        print(f"Synchronized worker thread started by {threading.current_thread().name}")        time.sleep(2)        print(f"Synchronized worker thread finished by {threading.current_thread().name}")thread1 = threading.Thread(target=worker, name="Thread 1")thread2 = threading.Thread(target=worker, name="Thread 2")synchronized_thread1 = threading.Thread(target=synchronized_worker, name="Synchronized Thread 1")synchronized_thread2 = threading.Thread(target=synchronized_worker, name="Synchronized Thread 2")print("Main thread started")thread1.start()thread2.start()synchronized_thread1.start()synchronized_thread2.start()thread1.join()thread2.join()synchronized_thread1.join()synchronized_thread2.join()print("Main thread finished")

在这个示例中,主线程启动了多个普通线程和带锁的线程。主线程会等待所有线程都完成后再继续执行。注意,在实际编程中,需要根据具体情况使用适当的线程同步机制来保护共享资源。

上一篇:

日妆巨头均声称安全 部分消费者担忧不减

下一篇:

最后一页

x 广告
相关阅读