生产者消费者模型(能看懂文字就能明白系列)

2024-09-07 18:12

本文主要是介绍生产者消费者模型(能看懂文字就能明白系列),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

系列文章目录

能看懂文字就能明白系列
C语言笔记传送门
Java笔记传送门
🌟 个人主页:古德猫宁-

🌈 信念如阳光,照亮前行的每一步


前言

本节目标:

  1. 理解什么是阻塞队列,阻塞队列与普通队列的区别
  2. 理解什么是生产者消费者模型
  3. 生产者消费者模型的主要作用

一、阻塞队列

阻塞独立是一个特殊的队列,它具有以下特点:

  1. 线程安全
  2. 带有阻塞特性:即如果队列为空,这时继续出队列的话,会发生阻塞,阻塞到其他线程往队列添加元素为止。如果队列为满,这时继续入队列的话,也会发生阻塞,阻塞到其他线程从队列取出元素为止。

利用上述特性,我们可以使用阻塞队列来实现一个“生产者消费者模型“

二、什么是生产者消费者模型

举个生活中的例子,比如有A B C三个人,每个人分别负责做饺子皮,包饺子,但是做饺子皮的工具只有一个,A在做饺子皮的时候,别人只能看着A干活,这时候效率是不高的,我们可以让A专门只负责做饺子皮这一工作,B和C两人负责包饺子,这种干活模式就不会设计到“工具”的竞争,这时候A就会不停的生产饺子皮,B和C会不停的消耗饺子皮。

而A生产出的饺子皮必须要有一个地方存放着,这个“地方”就叫做阻塞队列。

A就是生产者,把生产出来的内容放到阻塞队列中。

B和C就是消费者,会从阻塞队列中获取内容并消费。

如果A生产的比较慢,B和C就得等待(因为他们从“空的队列”中获取元素就会阻塞)

如果A生产的比较快,A就得等待(因为往“满的队列”中添加元素也会阻塞)

三、生产者消费者模型的意义

  1. 解耦合:两个模块,联系越紧密,耦合就越高,尤其是对于分布式系统来说,是更加有意义的。
    在这里插入图片描述
    如上图所示,如果服务器A和服务器B直接交互(A把请求发送给B,B把响应返回A),这样彼此之间的耦合是比较高的。
    如果B出现问题,A很可能也会被影响到。
    如果未来再添加一个服务器C,就需要对服务器A这边的代码做出一定的改动。

相比之下,如果采用生产者消费者模型,可以有效解决这种耦合问题
在这里插入图片描述
此时,耦合就会被降低,如果服务器B这边出现问题,就不会对服务器A产生直接影响.(服务器A只是和队列交互,不知道服务器B的存在)
后续如果新增一个服务器C,此时,服务器A不用进行任何修改,只需要让服务器C从队列中获取数据即可.

  1. 削峰填谷
    这里的峰和谷:
    峰:短时间内请求量比较大
    谷:请求量比较小
    在这里插入图片描述
    在这个结构下,一旦客户端这边发起的请求非常多了,服务器A收到的每一个请求,都会立即发给服务器B,服务器A这边抗多少访问量,服务器B和服务器A完全一样。
    不同的服务器,上面跑到业务不同,虽然访问量一样,单个访问,消耗的硬件资源不一样,可能服务器A承担这些并发量没啥事,服务器B承担这些并发量就会挂了。(比如数据库本身就是一个分布式系统,相对脆弱)

引入生产者消费者模型,上述问题也可以被改善。
在这里插入图片描述
服务器A这边收到了较大的请求量,服务器A会把对应的请求写入到队列中,服务器B仍然可以按照之前的节奏,来处理请求。

比如,正常情况下,服务器A和服务器B每秒钟处理1000次请求,极端情况下,服务器A这边每秒要处理3000次请求,如果让服务器B也处理3000次,服务器B就要挂了,阻塞队列帮服务器B承担了压力,服务器B仍然可以按照1000次的节奏处理请求。

像上述的峰值情况,一般不会持续存在,只会短时间出现,过了峰值之后,服务器A的请求量就恢复正常了,服务器B就可以逐渐把挤压的数据都给处理掉了。

四、代码实现

public class blockingQueue {private String[] data = new String[1000];private volatile int head = 0;private volatile int rear = 0;private volatile int size = 0;public void put(String elem) throws InterruptedException {synchronized (this) {while (size == data.length) {//队列满了//如果是普通的队列,满了就直接return即可this.wait();}//队列没满,往里面添加元素data[rear] = elem;rear++;if (rear == data.length) {rear = 0;//如果rear自增之后,到达了队列结尾,这个时候需要让他回到开头(环形队列)}size++;//这个notify用来唤醒take中的waitthis.notify();}}public String take() throws InterruptedException {synchronized (this) {while (size == 0) {//队列空了//对于普通队列,直接返回即可this.wait();}//队列不空,就可以把对首元素(head位置的元素)删除掉,并进行返回String ret = data[head];head++;if (head == data.length) {head = 0;}size--;//这个notify用来唤醒put中的wait,表示当前队列不是满的,可以继续添加元素了this.notify();return ret;}}public static void main(String[] args) {blockingQueue queue = new blockingQueue();//消费者Thread t1 = new Thread(() -> {while (true) {try {String result = queue.take();System.out.println("消费元素:" + result);} catch (InterruptedException e) {e.printStackTrace();}}});//生产者Thread t2 = new Thread(()->{int num = 1;while (true){try {queue.put(num+" ");} catch (InterruptedException e) {e.printStackTrace();}num++;System.out.println("生产元素:"+num);try {Thread.sleep(500);} catch (InterruptedException e) {e.printStackTrace();}}});t1.start();t2.start();}
}

一个队列,要么是空,要么是满,take和put只有一边能阻塞,如果put阻塞了,其他线程调用put也都会阻塞,只有take唤醒,如果take阻塞了,其他线程继续调用take也还是会阻塞,只有靠put唤醒。


各位大佬点点关注点点赞

这篇关于生产者消费者模型(能看懂文字就能明白系列)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1145788

相关文章

0基础租个硬件玩deepseek,蓝耘元生代智算云|本地部署DeepSeek R1模型的操作流程

《0基础租个硬件玩deepseek,蓝耘元生代智算云|本地部署DeepSeekR1模型的操作流程》DeepSeekR1模型凭借其强大的自然语言处理能力,在未来具有广阔的应用前景,有望在多个领域发... 目录0基础租个硬件玩deepseek,蓝耘元生代智算云|本地部署DeepSeek R1模型,3步搞定一个应

Deepseek R1模型本地化部署+API接口调用详细教程(释放AI生产力)

《DeepseekR1模型本地化部署+API接口调用详细教程(释放AI生产力)》本文介绍了本地部署DeepSeekR1模型和通过API调用将其集成到VSCode中的过程,作者详细步骤展示了如何下载和... 目录前言一、deepseek R1模型与chatGPT o1系列模型对比二、本地部署步骤1.安装oll

Spring AI Alibaba接入大模型时的依赖问题小结

《SpringAIAlibaba接入大模型时的依赖问题小结》文章介绍了如何在pom.xml文件中配置SpringAIAlibaba依赖,并提供了一个示例pom.xml文件,同时,建议将Maven仓... 目录(一)pom.XML文件:(二)application.yml配置文件(一)pom.xml文件:首

如何在本地部署 DeepSeek Janus Pro 文生图大模型

《如何在本地部署DeepSeekJanusPro文生图大模型》DeepSeekJanusPro模型在本地成功部署,支持图片理解和文生图功能,通过Gradio界面进行交互,展示了其强大的多模态处... 目录什么是 Janus Pro1. 安装 conda2. 创建 python 虚拟环境3. 克隆 janus

本地私有化部署DeepSeek模型的详细教程

《本地私有化部署DeepSeek模型的详细教程》DeepSeek模型是一种强大的语言模型,本地私有化部署可以让用户在自己的环境中安全、高效地使用该模型,避免数据传输到外部带来的安全风险,同时也能根据自... 目录一、引言二、环境准备(一)硬件要求(二)软件要求(三)创建虚拟环境三、安装依赖库四、获取 Dee

DeepSeek模型本地部署的详细教程

《DeepSeek模型本地部署的详细教程》DeepSeek作为一款开源且性能强大的大语言模型,提供了灵活的本地部署方案,让用户能够在本地环境中高效运行模型,同时保护数据隐私,在本地成功部署DeepSe... 目录一、环境准备(一)硬件需求(二)软件依赖二、安装Ollama三、下载并部署DeepSeek模型选

Golang的CSP模型简介(最新推荐)

《Golang的CSP模型简介(最新推荐)》Golang采用了CSP(CommunicatingSequentialProcesses,通信顺序进程)并发模型,通过goroutine和channe... 目录前言一、介绍1. 什么是 CSP 模型2. Goroutine3. Channel4. Channe

Python基于火山引擎豆包大模型搭建QQ机器人详细教程(2024年最新)

《Python基于火山引擎豆包大模型搭建QQ机器人详细教程(2024年最新)》:本文主要介绍Python基于火山引擎豆包大模型搭建QQ机器人详细的相关资料,包括开通模型、配置APIKEY鉴权和SD... 目录豆包大模型概述开通模型付费安装 SDK 环境配置 API KEY 鉴权Ark 模型接口Prompt

Spring Security 从入门到进阶系列教程

Spring Security 入门系列 《保护 Web 应用的安全》 《Spring-Security-入门(一):登录与退出》 《Spring-Security-入门(二):基于数据库验证》 《Spring-Security-入门(三):密码加密》 《Spring-Security-入门(四):自定义-Filter》 《Spring-Security-入门(五):在 Sprin

大模型研发全揭秘:客服工单数据标注的完整攻略

在人工智能(AI)领域,数据标注是模型训练过程中至关重要的一步。无论你是新手还是有经验的从业者,掌握数据标注的技术细节和常见问题的解决方案都能为你的AI项目增添不少价值。在电信运营商的客服系统中,工单数据是客户问题和解决方案的重要记录。通过对这些工单数据进行有效标注,不仅能够帮助提升客服自动化系统的智能化水平,还能优化客户服务流程,提高客户满意度。本文将详细介绍如何在电信运营商客服工单的背景下进行