Python的线程之线程同步

线程同步是指在多线程并发执行的场景中,保证各个线程协作正确可靠的一种机制。在Python中,我们通常使用锁(Lock)和条件变量(Condition)两种机制进行线程同步。

一、锁(Lock)

1.1 什么是锁

锁是一种线程同步机制,主要用于协调多个线程的并发访问,实现线程之间的互斥。

1.2 锁的实现机制

Python中的锁是通过Lock对象实现的,Lock的状态只有两种:锁定和未锁定。每当一个线程将一个锁对象锁定后,其他线程就无法再次获取该锁,直到该锁被锁定的线程释放锁。

1.3 使用锁

在Python中,使用锁可以非常简单的实现线程同步。示例代码如下:

import threading

# 创建锁
lock = threading.Lock()

def work():
    # 获取锁
    lock.acquire()
    try:
        for i in range(5):
            print("{} is running...".format(threading.current_thread().name))
    finally:
        # 释放锁
        lock.release()

if __name__ == '__main__':
    # 创建两个线程
    t1 = threading.Thread(target=work)
    t2 = threading.Thread(target=work)

    # 启动线程
    t1.start()
    t2.start()

在上述示例中,我们创建了两个线程,它们的工作函数是打印5次当前线程的名字。由于这两个线程都需要输出,因此需要使用锁机制进行线程同步。

首先,我们使用threading.Lock()创建一个锁对象。然后在工作函数中,使用lock.acquire()获取锁,表示锁被占用。在获取锁后,线程可以执行自己的任务。

最后,使用lock.release()释放锁,表示任务执行完毕,锁被释放。只有当锁被释放后,其他线程才能获取锁,执行自己的任务。

二、条件变量(Condition)

2.1 什么是条件变量

条件变量是一个线程同步机制,它的主要作用是在多个线程之间协调共享数据的访问。

2.2 条件变量的实现机制

在Python中,条件变量是通过Condition对象实现的。Condition对象内部有一个锁对象,它通过wait()notify()notify_all()三个方法协调线程之间的协作。

wait(): 等待条件变量。当条件变量被唤醒时,会重新获取锁,并返回True。

notify(n=1): 唤醒指定数量的线程,使得它们从等待队列中出队,但不会立即释放锁。

notify_all(): 唤醒所有等待的线程。

2.3 使用条件变量

示例1:

import threading

class Producer(threading.Thread):
    def __init__(self, cond, name):
        super().__init__(name=name)
        self.cond = cond

    def run(self):
        with self.cond:
            print("生产者已准备好。")
            self.cond.notify()

class Consumer(threading.Thread):
    def __init__(self, cond, name):
        super().__init__(name=name)
        self.cond = cond

    def run(self):
        with self.cond:
            print("等待生产者准备中...")
            self.cond.wait()
            print("消费者已开始消费。")

if __name__ == "__main__":
    cond = threading.Condition()
    p = Producer(cond, "Producer")
    c = Consumer(cond, "Consumer")
    p.start()
    c.start()
    p.join()
    c.join()

在上面的示例中,我们使用条件变量协调了两个线程的协作。在Producer线程中,我们通过notify()方法唤醒了Consumer线程,并允许其执行任务。在Consumer线程中,我们通过wait()方法等待生产者准备就绪,并等待被唤醒通知。

示例2:

import threading

MAX_NUM = 10

class Producer(threading.Thread):
    def __init__(self, cond, name):
        super().__init__(name=name)
        self.cond = cond

    def run(self):
        with self.cond:
            for i in range(MAX_NUM):
                # 生产
                print("{} 生产了 {}".format(self.name, i))

                # 控制队列长度
                if i == MAX_NUM - 1:
                    self.cond.notify_all()
                    self.cond.wait()
                else:
                    self.cond.notify()

class Consumer(threading.Thread):
    def __init__(self, cond, name):
        super().__init__(name=name)
        self.cond = cond

    def run(self):
        with self.cond:
            for i in range(MAX_NUM):
                self.cond.wait()
                # 消费
                print("{} 消费了 {}".format(self.name, i))
                self.cond.notify()

if __name__ == "__main__":
    cond = threading.Condition()
    p = Producer(cond, "Producer")
    c1 = Consumer(cond, "Consumer1")
    c2 = Consumer(cond, "Consumer2")
    p.start()
    c1.start()
    c2.start()
    p.join()
    c1.join()
    c2.join()

在上面的示例中,我们使用条件变量模拟了一个进程间通信的生产者-消费者模型。在Producer线程中,我们每次生成一个数据并使用notify()方法通知一个消费者。在最后一次生成一个数据时,我们使用notify_all()方法,将所有等待中的消费者都唤醒并等待生产者再次notify()。在Consumer线程中,我们首先使用wait()方法等待生产者的通知,随后打印信息并使用notify()方法通知生产者继续往队列中生产。

本站文章如无特殊说明,均为本站原创,如若转载,请注明出处:Python的线程之线程同步 - Python技术站

(0)
上一篇 2023年5月19日
下一篇 2023年5月19日

相关文章

  • python监控文件或目录变化

    Python 监控文件或目录变化通常使用第三方库 watchdog。以下是完整的攻略: 安装watchdog 在终端中执行以下命令进行安装: pip install watchdog 监控文件的变化 以下是监控文件变化的示例代码: import time from watchdog.observers import Observer from watchdo…

    python 2023年6月3日
    00
  • 简单讲解Python编程中namedtuple类的用法

    当我们需要定义一些复杂的数据类型时,可以使用Python中的namedtuple类。namedtuple是一个Python标准库集合模块中的数据类型,它是一个高性能的tuple子类,它允许定义带有命名字段的元组,元组内的每个元素都可以通过名称和索引访问。 下面是namedtuple类用法的详细说明: 什么是namedtuple namedtuple是Pyth…

    python 2023年5月14日
    00
  • Python多线程编程(八):使用Event实现线程间通信

    我们来详细讲解一下Python多线程编程中使用Event实现线程间通信的完整攻略。 什么是Event? Event是Python中内置的一个线程同步机制,它是一种简单的线程间通信方式。在多个线程之间,一个线程可以通过设置Event来通知其他线程,其他线程也可以通过检查Event的状态来判断是否有通知需要处理。 Event的使用方法 在使用Event时,一般需…

    python 2023年5月19日
    00
  • Linux下查看nginx apache mysql php的编译参数

    可以使用以下步骤在Linux下查看nginx、apache、mysql和php的编译参数: 查看nginx的编译参数 进入nginx的安装目录,在bin下找到可执行文件nginx,使用以下命令获取nginx的编译参数: /usr/local/nginx/sbin/nginx -V 示例输出: “` nginx version: nginx/1.18.0 b…

    python 2023年6月3日
    00
  • 一文搞懂Python读取text,CSV,JSON文件的方法

    针对“一文搞懂Python读取text,CSV,JSON文件的方法”的完整攻略,我将分为以下几个部分进行讲解: 读取text文件的方法 读取CSV文件的方法 读取JSON文件的方法 示例说明 1. 读取text文件的方法 在Python中,我们可以使用built-in函数open()来读取text文件,并使用read()方法来读取文件内容。代码示例如下: w…

    python 2023年6月3日
    00
  • Android 中 EventBus 的使用之多线程事件处理

    Android 中 EventBus 的使用之多线程事件处理 什么是 EventBus? EventBus 是一种 Android 常用的事件发布/订阅框架,其可以使用简单的发布/订阅机制来简化应用程序的组件之间的通信,而不需要使用复杂或模糊的 BroadcastReceiver,而且 EventBus 完全基于 Java 的发布/订阅模式的实现。它使组件之…

    python 2023年6月13日
    00
  • Python多层嵌套list的递归处理方法(推荐)

    以下是详细讲解“Python多层嵌套list的递归处理方法(推荐)”的完整攻略。 在Python中,多层嵌套的列表(list)是一种常见的数据结构。在处理多层套的列表时,可以使用递归的方法来遍历和处理列表中的元素。下面是一些常见的递归处理方法。 方法一:使用递归函数 def process_list(lst): for item in lst: if isi…

    python 2023年5月13日
    00
  • 16异常处理

    异常处理 异常 异常即是一个事件,该事件会在程序执行过程中发生,影响了程序的正常执行。 一般情况下,在Python无法正常处理程序时就会发生一个异常。 异常是Python对象,表示一个错误。 当Python脚本发生异常时我们需要捕获处理它,否则程序会终止执行。 捕获异常 异常类型捕获 # 捕获常规异常 try: 可能发生错误的代码 except: 如果出行异…

    python 2023年4月27日
    00
合作推广
合作推广
分享本页
返回顶部