Python的线程之线程同步

yizhihongxing

线程同步是指在多线程并发执行的场景中,保证各个线程协作正确可靠的一种机制。在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中request发送post请求传递json参数的问题

    以下是关于“关于Python中request发送post请求传递json参数的问题”的完整攻略: 关于 Python 中 request 发送 post 请求传递 json 参数的问题 在 Python 中,我们可以使用 requests 模块发送 HTTP 请求。当我们需要发送 POST 请求并传递 JSON 参数时,我们可以使用 requests 模块的…

    python 2023年5月15日
    00
  • Python装饰器实现方法及应用场景详解

    Python装饰器实现方法及应用场景详解 1. 概述 装饰器是 Python 中非常重要的概念,几乎所有 Python 框架都大量使用到了装饰器。它可以用于功能增强、日志处理、输入验证和安全控制等场景。 装饰器本质上是一个 Python 函数或类,并在不改变原函数/方法定义的基础上对其进行增强。Python 中借助函数式编程的特点,可以很方便地实现装饰器。 …

    python 2023年6月3日
    00
  • 使用python-cv2实现视频的分解与合成的示例代码

    这里是使用 Python 的 OpenCV 库(cv2)实现视频的分解与合成的完整攻略。 准备工作 在开始之前,需要先安装 OpenCV 库才能运行这项任务。可以通过 pip install 命令安装: pip install opencv-python 视频的读取 首先,让我们看一个从视频中读取每一帧并显示的示例: import cv2 # 加载视频源 v…

    python 2023年5月19日
    00
  • Python 相对路径和绝对路径及写法演示

    下面是 Python 相对路径和绝对路径的详细讲解以及写法演示攻略。 相对路径和绝对路径 在 Python 中,文件的路径有两种表示方式:相对路径和绝对路径。相对路径是相对于当前文件所在的目录的路径,而绝对路径则是从文件系统的根目录开始的路径。 使用相对路径可以更方便地描述文件的位置,但是相对路径存在依赖关系,修改文件结构可能会导致代码出现运行错误。而绝对路…

    python 2023年6月2日
    00
  • python中reader的next用法

    Python中的csv模块是处理CSV(Comma Separated Value,逗号分隔值)数据文件的常用工具。它的reader类提供了一系列方法用于逐行读取CSV文件中的数据。其中,next()方法可以读取并返回文件的下一行数据,返回值为一个列表,列表中包含了该行数据的各个字段。 下面是next()方法的基本使用方式: import csv # 打开C…

    python 2023年6月3日
    00
  • Python大数据量文本文件高效解析方案代码实现全过程

    处理大数据量文本文件是数据分析和处理中的常见任务。Python提供了多种高效的解析方案,包括使用pandas、numpy和内置的文件操作函数等。以下是详细讲解Python大数据量文本文件高效解析方案的攻略,包含两个例。 示例1:使用pandas解析CSV文件 以下是一个示例,可以使用pandas解析CSV文件: import pandas as pd # 读…

    python 2023年5月15日
    00
  • python爬虫爬取指定内容的解决方法

    当我们需要快速收集大量需要的数据时,Python爬虫就是一个非常有用的工具。Python爬虫具有快速、高效、灵活等优势,并且非常适合于大规模数据采集。在使用Python爬虫时,我们最常见的需求之一是需要只爬取指定内容。下面是详细的攻略过程: 步骤一:查找指定内容的来源 首先,查找指定内容的来源。有可能这些内容都在某一特定网站或某一特定页面中,如果我们能确定这…

    python 2023年5月14日
    00
  • Python数据分析与处理(一)–北京高考分数线统计分析

    下面我将详细讲解以下这篇文章的内容: Python数据分析与处理(一)–北京高考分数线统计分析 1. 引言 本教程主要介绍如何使用 Python 进行数据处理与分析,以北京市高考分数线为例子,介绍 Python 中 Pandas、Matplotlib 等常用数据分析工具的使用,通过实例展示如何对数据进行统计分析、可视化呈现。同时,本教程也是一个完整的实例教…

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