探索生产者/消费者模式:解决并发编程中的资源竞争

序言

在并发编程中,资源竞争是一个常见的问题。为了有效地管理资源并确保线程安全,需要采用一些有效的方法。其中之一是生产者/消费者模式,它是一种经典的并发设计模式,用于解决生产者和消费者之间的协作问题。本文将深入探讨生产者/消费者模式的概念、应用场景以及实现方法。

一、什么是生产者/消费者模式

生产者/消费者模式是一种并发设计模式,用于解决多线程环境下的资源共享与同步问题。它涉及两种类型的线程:生产者和消费者。生产者负责生成数据或任务,并将它们放入共享的缓冲区中,而消费者则负责从缓冲区中取出数据或任务并进行处理。

二、应用场景

生产者/消费者模式在许多实际场景中都有广泛的应用,其中包括但不限于:

  1. 生产者/消费者问题:在计算机科学中,生产者/消费者问题是一个经典的问题,涉及到多个生产者和消费者并共享一个有限大小的缓冲区。生产者生成数据并将其放入缓冲区,消费者则从缓冲区中取出数据并进行处理。生产者和消费者之间必须进行同步,以确保缓冲区不会溢出或下溢。
  2. 线程池:线程池是一种常见的并发编程模式,用于管理和复用线程。生产者负责将任务提交给线程池,而线程池中的线程则充当消费者,负责执行这些任务。
  3. 事件驱动编程:在事件驱动编程中,事件生成者(生产者)生成事件并将其放入事件队列中,而事件处理程序(消费者)则从队列中取出事件并处理它们。

四、实现方法

在实现生产者/消费者模式时,有几种常见的方法:

  1. 使用线程和共享缓冲区:这是最直接的实现方法。生产者线程生成数据并将其放入共享的缓冲区,而消费者线程则从缓冲区中取出数据。需要注意的是,对于共享缓冲区的访问需要进行同步,以防止竞争条件和数据不一致性。
  2. 使用阻塞队列:阻塞队列是一种线程安全的队列数据结构,它支持在队列为空时阻塞消费者线程,并在队列已满时阻塞生产者线程。通过使用阻塞队列,可以简化生产者/消费者模式的实现,并减少竞争条件的发生。
  3. 使用信号量:信号量是一种并发原语,用于控制对共享资源的访问。生产者和消费者可以使用信号量来进行同步,以确保缓冲区的访问不会发生冲突。

五、使用案例

场景假设:我们正在开发一个在线商店系统,其中有一个订单管理模块。在这个模块中,订单被创建并放入一个订单队列,然后由后台工作人员逐个处理

  1. 定义实体类:表示订单

    @Data
    @NoArgsConstructor
    @AllArgsConstructor
    public class Order {
        private int orderId; // 订单 id
        private String customerName; // 顾客姓名
    }
    
  2. 定义生产者:不断生成新的订单

    public class OrderProducer implements Runnable {
    
        private final Queue<Order> orderQueue; // 订单队列
        private final int maxQueueSize; // 订单队列最大容量
        private int orderNumber; // 订单编号
    
        public OrderProducer(Queue<Order> orderQueue, int maxQueueSize) {
            this.orderQueue = orderQueue;
            this.maxQueueSize = maxQueueSize;
            // 初始订单编号为 1
            this.orderNumber = 1;
        }
    
        @Override
        public void run() {
            while (true) {
                synchronized (orderQueue) {
                    // 当订单队列满时等待
                    while (orderQueue.size() == maxQueueSize) {
                        try {
                            System.out.println("Order queue is full, waiting for orders to be processed...");
                            // 等待订单队列有空间
                            orderQueue.wait();
                        } catch (InterruptedException e) {
                            System.out.println("处理生产者异常");
                        }
                    }
    
                    // 创建新订单并添加到订单队列中
                    Order order = new Order(orderNumber++, "Customer " + orderNumber);
                    orderQueue.add(order);
                    System.out.println("New order added: Order #" + order.getOrderId() + " by " + order.getCustomerName());
                    // 通知订单处理者有新订单
                    orderQueue.notifyAll();
                }
            }
        }
    }
    
  3. 定义消费者:不断处理新订单

    public class OrderConsumer implements Runnable {
        // 订单队列
        private final Queue<Order> orderQueue;
    
        public OrderConsumer(Queue<Order> orderQueue) {
            this.orderQueue = orderQueue;
        }
    
        @Override
        public void run() {
            while (true) {
                synchronized (orderQueue) {
                    // 当订单队列为空时等待
                    while (orderQueue.isEmpty()) {
                        try {
                            System.out.println("No orders in the queue, waiting for new orders...");
                            // 等待新订单
                            orderQueue.wait();
                        } catch (InterruptedException e) {
                            System.out.println("处理消费者异常");
                        }
                    }
    
                    // 从订单队列中取出订单并处理
                    Order order = orderQueue.poll();
                    System.out.println("Order processed: Order #" + order.getOrderId() + " by " + order.getCustomerName());
                    // 通知订单生产者有空间
                    orderQueue.notifyAll();
                }
            }
        }
    }
    
  4. 测试生产者/消费者模式

    public static void main(String[] args) {
        Queue<Order> orderQueue = new LinkedList<>(); // 订单队列
        int maxQueueSize = 10; // 订单队列最大容量
    
        OrderProducer orderProducer = new OrderProducer(orderQueue, maxQueueSize); // 订单生产者
        OrderConsumer orderProcessor = new OrderConsumer(orderQueue); // 订单处理者
    
        Thread producerThread = new Thread(orderProducer); // 订单生产者线程
        Thread processorThread = new Thread(orderProcessor); // 订单处理者线程
    
        producerThread.start(); // 启动订单生产者线程
        processorThread.start(); // 启动订单处理者线程
    }
    

    测试效果:

    image.png

六、FAQ

生产者/消费者模式是一种有效的并发设计模式,用于解决资源共享与同步的问题。通过合理地设计和实现生产者和消费者之间的协作,可以提高系统的性能和可靠性,同时减少竞争条件和数据不一致性的发生。在实际应用中,可以根据具体的场景选择合适的实现方法,并结合其他并发编程技术来构建高效、可靠的并发系统。

推荐阅读

  1. 深入探究 Spring Boot Starter:从概念到实践
  2. RBAC 权限设计(五)
  3. Docker Compose:简化多容器应用部署
  4. cURL:命令行下的网络工具
  5. RabbitMQ(Docker 单机部署)

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

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

相关文章

sentinel-dashborard下发流控规则未生效排查解决

Sentinel简介 Sentinel是阿里巴巴开源的一个流量控制组件&#xff0c;它以流量为切入点&#xff0c;从流量控制、熔断降级、系统负载保护等多个维度来保护服务的稳定性。Sentinel的核心思想是&#xff1a;通过动态设置的规则&#xff0c;对进入系统的流量进行控制&#xff0c;…

【好困】磁场里的瞌睡虫:地磁暴真的会让我们感到疲倦吗?

【好困】磁场里的瞌睡虫&#xff1a;地磁暴真的会让我们感到疲倦吗&#xff1f; 写在最前面地磁暴真的会让我们感到疲倦吗&#xff1f;一探究竟地磁暴是什么&#xff1f;地磁暴如何影响人体&#xff1f;结论 &#x1f308;你好呀&#xff01;我是 是Yu欸 &#x1f30c; 2024每…

工业机器人应用实践之玻璃涂胶(篇二)

工业机器人 接上篇文章&#xff0c;浅谈一下实践应用&#xff0c;具体以玻璃涂胶为例&#xff1a; 了解工业机器人在玻璃涂胶领域的应用认识工具坐标系的标定方法掌握计时指令的应用掌握人机交互指令的应用掌握等待类指令用法&#xff08;WaitDI、WaitUnitl 等&#xff09;认…

基于springboot+vue+Mysql的外卖点餐系统

开发语言&#xff1a;Java框架&#xff1a;springbootJDK版本&#xff1a;JDK1.8服务器&#xff1a;tomcat7数据库&#xff1a;mysql 5.7&#xff08;一定要5.7版本&#xff09;数据库工具&#xff1a;Navicat11开发软件&#xff1a;eclipse/myeclipse/ideaMaven包&#xff1a;…

聚合工程搭建、工程依赖导入

上一章讲了自动化云测平台的一些环境的准备 如果还未完成云服务器的环境搭建和本地环境的搭建&#xff0c;请点击左侧 -> 传送门 那么正式开始这一章的内容 聚合工程搭建 创建项目 我们先给项目命个名&#xff1a;xxx-meter&#xff0c;构建系统&#xff0c;我们选择M…

分布式事务?哪几种方式实现?一文看懂!

什么是分布式事务 分布式事务是指在分布式系统中涉及到多个数据库或多个应用程序之间的事务处理&#xff0c;这些数据库或应用程序可能分布在不同的物理节点上&#xff0c;甚至可能位于不同的地理位置。在分布式事务中&#xff0c;需要确保所有参与者的事务操作都能够保持一致性…

Windows Nginx 服务器部署并推流

环境 开发环境&#xff1a;windows 开发工具&#xff1a;ffmpeg、nginx、nginx-rmtp-module、vlc media player Nginx Nginx是一个高性能的HTTP和反向代理web服务器&#xff0c;并且支持rtmp&#xff0c;不过rtmp是作为一个nginx的一个模块。 对于linux系统&#xff0c;需要另…

IIS配置SSL,根据pem和key生成pfx,openssl的版本不能太高

1、生成pfx文件 供应商给的文件是pef和key后缀的两个文件&#xff0c;在IIS里不好导入(如果有知道好导入的可以给我留言&#xff0c;谢谢。)。 1.1 下载OpenSSL工具&#xff0c;并安装。 主要用于将.pem文件转成.pfx文件。 下载OpenSSL的链接&#xff1a;http://slproweb.com/…

【RAG 博客】RAG 应用中的 Routing

Blog&#xff1a;Routing in RAG-Driven Applications ⭐⭐⭐⭐ 根据用户的查询意图&#xff0c;在 RAG 程序内部使用 “Routing the control flow” 可以帮助我们构建更实用强大的 RAG 程序。路由模块的关键实现就是一个 Router&#xff0c;它根据 user query 的查询意图&…

VTK官方示例

VTK官方示例 -vtk字體 #!/usr/bin/env python# noinspection PyUnresolvedReferences import vtkmodules.vtkInteractionStyle # noinspection PyUnresolvedReferences import vtkmodules.vtkRenderingFreeType # noinspection PyUnresolvedReferences import vtkmodules.vtk…

【MySQL】基本操作

欢迎来到Cefler的博客&#x1f601; &#x1f54c;博客主页&#xff1a;折纸花满衣 &#x1f3e0;个人专栏&#xff1a;MySQL 目录 &#x1f449;&#x1f3fb;创建和删除数据库&#x1f449;&#x1f3fb;数据库编码集和数据库校验集校验规则对数据库的影响 &#x1f449;&…

GT2505HS-VTBD 三菱触摸屏手持式5.7寸型

GT2505HS-VTBD 三菱触摸屏手持式5.7寸型 GT2505HS-VTBD参数,GT2505HS-VTBD用户手册,GT2505HS-VTBD使用手册GT2505HS-VTBD参数说明&#xff1a;手持式5.7吋型&#xff0c;VGA 640*480&#xff0c;TFT彩色液晶屏,65536色,内存32MB&#xff0c;DC24V,内置以太网接口。 GT2505HS-VT…

【GlobalMapper精品教程】079:投影坐标系转地理坐标系(UTM转WGS1984/2000)

文章目录 一、矢量UTM转WGS1984/20001. UTM转WGS19842. UTM转CGCS2000二、栅格UTM转WGS1984/2000一、矢量UTM转WGS1984/2000 加载配套实验数据(data079.rar)中的矢量数据,如下所示: 查看源坐标系:双击图层的,图层投影选项卡,为UTM投影,Zone48N。 设置系统坐标系:点击…

jupyter_lab修改默认目录

1、新版本的jupyter如果按照网上说的只修改jupyter_notebook_config.py这个文件没啥用&#xff0c;我在实际修改过程中遇到了如下报错。 经过各种尝试&#xff0c;还需要修改jupyter_notebook_config.py这个配置文件 2、不废话&#xff0c;直接上步骤 在Jupyter Notebook或者c…

2024年10大AI动画工具

在当今快节奏的数字环境中&#xff0c;动画师和内容创作者不断寻求创新工具来提高他们的工作效率和创造力。随着人工智能的出现&#xff0c;动画发生了显着的转变&#xff0c;因此提供了大量的选项&#xff0c;使图像无缝地栩栩如生。 无论你是一位有抱负的动画师还是经验丰富…

#include《初见C语言之顺序表的增删查改》

目录 一、顺序表 二、顺序表的分类 三、顺序表的实现前期准备 第一步&#xff0c;确定需要的文件 第二步&#xff0c;开始分析 四、顺序表的实现 1.初始化 2.销毁 3.申请空间 4.打印 5.尾插 6.头插 7.尾删 8.头删 9.指定位置之前插入 10.指定位置之前删除 11.…

grid的常见使用场景

场景1&#xff1a;固定几列显示&#xff0c;显示不下会自动换行 <div id"container"><div class"item item-1">1</div><div class"item item-2">2</div><div class"item item-3">3</div>&l…

vscode go语言开发中在任意包运行和调试代码 Example使用方法

一般情况下我们在进行go语言开发的时候我们都需要创建一个main方法和main包才能运行go代码&#xff0c; 针对这个问题&#xff0c;go语言给我们内置了功能强大的testing测试框架&#xff0c; 其中一个很有意思的Example测试就非常的方便使用。 他不管你在什么包&#xff0c;也…

LNMP环境部署WordPress——使用源码包安装方式部署环境

目录 一.前提准备 二.源码安装Mysql 1.MySQL类型 2.MySQL 版本说明 3.MySQL 安装方式 3.1 yum 安装 3.2 编译安装 3.3 二进制安装 3.4 rpm 安装 4. 编译安装MySQL5.7 4.1 清理安装环境 4.2 创建mysql用户 4.3 从官网下载tar包 4.4 安装编译工具 4.5 解压 4.6 …

Vue3:路由

1. 路由简介 在Vue3中&#xff0c;路由是一个核心概念&#xff0c;特别是在构建单页面应用程序&#xff08;SPA&#xff09;时。以下是Vue3中路由的基本概念&#xff1a; 1. **路由&#xff08;Route&#xff09;**&#xff1a;在Vue3中&#xff0c;路由是指根据特定的规则将用…