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

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

相关文章

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项目增添不少价值。在电信运营商的客服系统中,工单数据是客户问题和解决方案的重要记录。通过对这些工单数据进行有效标注,不仅能够帮助提升客服自动化系统的智能化水平,还能优化客户服务流程,提高客户满意度。本文将详细介绍如何在电信运营商客服工单的背景下进行

Andrej Karpathy最新采访:认知核心模型10亿参数就够了,AI会打破教育不公的僵局

夕小瑶科技说 原创  作者 | 海野 AI圈子的红人,AI大神Andrej Karpathy,曾是OpenAI联合创始人之一,特斯拉AI总监。上一次的动态是官宣创办一家名为 Eureka Labs 的人工智能+教育公司 ,宣布将长期致力于AI原生教育。 近日,Andrej Karpathy接受了No Priors(投资博客)的采访,与硅谷知名投资人 Sara Guo 和 Elad G

高效录音转文字:2024年四大工具精选!

在快节奏的工作生活中,能够快速将录音转换成文字是一项非常实用的能力。特别是在需要记录会议纪要、讲座内容或者是采访素材的时候,一款优秀的在线录音转文字工具能派上大用场。以下推荐几个好用的录音转文字工具! 365在线转文字 直达链接:https://www.pdf365.cn/ 365在线转文字是一款提供在线录音转文字服务的工具,它以其高效、便捷的特点受到用户的青睐。用户无需下载安装任何软件,只

Retrieval-based-Voice-Conversion-WebUI模型构建指南

一、模型介绍 Retrieval-based-Voice-Conversion-WebUI(简称 RVC)模型是一个基于 VITS(Variational Inference with adversarial learning for end-to-end Text-to-Speech)的简单易用的语音转换框架。 具有以下特点 简单易用:RVC 模型通过简单易用的网页界面,使得用户无需深入了

科研绘图系列:R语言扩展物种堆积图(Extended Stacked Barplot)

介绍 R语言的扩展物种堆积图是一种数据可视化工具,它不仅展示了物种的堆积结果,还整合了不同样本分组之间的差异性分析结果。这种图形表示方法能够直观地比较不同物种在各个分组中的显著性差异,为研究者提供了一种有效的数据解读方式。 加载R包 knitr::opts_chunk$set(warning = F, message = F)library(tidyverse)library(phyl

透彻!驯服大型语言模型(LLMs)的五种方法,及具体方法选择思路

引言 随着时间的发展,大型语言模型不再停留在演示阶段而是逐步面向生产系统的应用,随着人们期望的不断增加,目标也发生了巨大的变化。在短短的几个月的时间里,人们对大模型的认识已经从对其zero-shot能力感到惊讶,转变为考虑改进模型质量、提高模型可用性。 「大语言模型(LLMs)其实就是利用高容量的模型架构(例如Transformer)对海量的、多种多样的数据分布进行建模得到,它包含了大量的先验

图神经网络模型介绍(1)

我们将图神经网络分为基于谱域的模型和基于空域的模型,并按照发展顺序详解每个类别中的重要模型。 1.1基于谱域的图神经网络         谱域上的图卷积在图学习迈向深度学习的发展历程中起到了关键的作用。本节主要介绍三个具有代表性的谱域图神经网络:谱图卷积网络、切比雪夫网络和图卷积网络。 (1)谱图卷积网络 卷积定理:函数卷积的傅里叶变换是函数傅里叶变换的乘积,即F{f*g}