2024.2.25 模拟实现 RabbitMQ —— 网络通信设计(服务器)

目录

引言

约定应用层的通信协议

自定义应用层协议

Type

Length

PayLod

 实现 Broker Server 类

属性 与 构造

启动 Broker Server

停止 Broker Server

处理客户端连接

读取请求 与 写回响应

根据请求计算响应

清除 channel 


引言

  • 生产者 和 消费者 都是客户端,均通过 网络 和 Broker Server 进行通信

注意点一:

  • 此处我们将使用 TCP 协议来作为通信的底层协议

注意点二:

  • TCP 是有连接的(Connection)
  • 由于 创建/断开 TCP 连接的成本还挺高,需要三次握手啥的
  • 所以为了能够让 TCP 连接得到复用
  • 我们还将创建一个 Channel 类作为 Connection 内部的 逻辑上 的连接
  • 即一个 Connection 中可能有多个 Channel(一个管道,多个网线传输的效果)

约定应用层的通信协议

  • 此处要交互的 Message 为 二进制数据
  • HTTP 为文本协议,JSON 为文本格式,不适用此处场景
  • 所以我们自定义一个应用层协议,使用二进制的方式来传输数据!

自定义应用层协议


Type

  • type 描述当前这个请求或响应是干啥的

具体理解:

  • 在我们的 MQ 中,客户端(生产者 + 消费者)和 服务器(Broker Server)之间要进行的操作就是 VirtualHost 中的那些核心 API
  • 我们希望客户端通过网络能够远程调用 VirtualHost 中的核心 API
  • 此处 type 就是在描述当前这个请求/响应是在调用哪个 API
  • 取值如下:
  • 0x1 创建 channel
  • 0x2 关闭 channel
  • 0x3 创建 exchange
  • 0x4 销毁 exchange
  • 0x5 创建 queue
  • 0x6 销毁 queue
  • 0x7 创建 binding
  • 0x8 销毁 binding
  • 0x9 发送 message
  • 0xa 订阅 message
  • 0xb 返回 ack
  • 0xc 服务器给客户端推送消息(被订阅的消息)响应独有的

Length

  • length 用来描述 payload 长度(防止粘包问题)

PayLod

  • payload 会根据当前是请求还是响应,以及当前的 type 有不同的取值

实例理解


实例一:

  • 比如 type 是 0x3(创建交换机),同时当前是一个请求
  • 此时 payload 里的内容,就相当于是 exchangeDeclare 的参数序列化的结果

具体代码实现:

  • 按照上述自定义应用层协议 创建 Request 类
import lombok.Data;/*
* 表示一个网络通信中的请求对象,按照自定义协议的格式来展开的
* */
@Data
public class Request {private int type;private int length;private byte[] payload;
}
  • 按照上述自定义应用层协议 创建 BasicArguments 类用于表示各方法的公共参数
import lombok.Data;import java.io.Serializable;/*
* 使用这个类表示方法的公共参数/辅助的字段
* 后续使用每个方法又会有一些不同的参数,不同的参数再分别使用不同的子类来表示
* */
@Data
public class BasicArguments implements Serializable {
//    表示一次请求/响应 的身份标识,可以把请求和响应对上protected String rid;
//    这个通信使用的 channel 的身份标识protected String channelId;
}
  • 每个方法有不同的参数,此处实例 type = 0x3 ,即 创建交换机(exchangeDeclare
  • 所以我们根据 VirtualHost 中的 exchangeDeclare 方法中的参数,单独创建一个类出来
  • 该类还需 继承用于表示公共参数的 BasicArguments 类
import com.example.demo.mqserver.core.ExchangeType;
import lombok.Getter;
import lombok.Setter;import java.io.Serializable;
import java.util.Map;@Getter
@Setter
public class ExchangeDeclareArguments extends BasicArguments implements Serializable {private String exchangeName;private ExchangeType exchangeType;private boolean durable;private boolean autoDelete;private Map<String,Object> arguments;
}

注意:

  • 其他 type 类型(除 0x1、0x2 、0xa 外)也均根据 其在 VirtualHost 中对应的参数,单独创建一个类即可
  • 0x1 和 0x2 分别为 创建 channel 和 关闭 channel,二者 API 所需参数就是公共参数,使用 BasicArguments 类即可,无需单独创建类
  • type = 0xa,即 订阅消息(basicConsume),后文详细讲解

实例二:

  • 比如 type = 0x3(创建交换机),同时当前是一个响应
  • 此时 payload 里的内容,就是 exchangeDeclare 的返回结果的序列化内容

具体代码实现:

  • 按照上述自定义应用层协议 创建 Response 类
import lombok.Data;/*
* 这个对象表示一个响应,也是根据自定义应用层协议来的
* */
@Data
public class Response {private int type;private int length;private byte[] payload;
}
  • 按照上述自定义应用层协议 创建 BasicReturns 类用于表示远程调用方法的返回值
import lombok.Data;import java.io.Serializable;/*
* 这个类表示各个远程调用的方法的返回值和公共信息
* */
@Data
public class BasicReturns implements Serializable {
//    用来标识唯一的请求和响应protected String rid;
//    用来标识一个 channelIdprotected String channelId;
//    表示当前这个远程调用方法的返回值protected boolean ok;
}

注意:

  • 其他 type 类型(除 0xc 外)均使用 BasicReturns 类中的成员变量 作为返回参数
  • type = 0xc,该 type 类型为响应独占,表示 服务器给客户端推送消息(被订阅的消息),后文详解讲解

特例一:

  • 比如 type = 0xa(订阅消息),同时当前是一个请求
  • 这个核心 API 比较特殊,其参数中包含有 回调函数

具体代码编写:

  • 我们根据 VirtualHost 中的 BasicConsume 方法中的参数,单独创建一个类出来
  • 并且该类也要 继承用于表示公共参数的 BasicArguments 类
  • 唯一不同的是,其中用于表示 回调函数的参数 consumer 我们不写入该类中
  • 也就代表着在客户端发送请求时,不再携带 consumer 参数
  • 因为在 broker server 这边,我们规定 BasicConsume 的回调方法统一为 将收到的消息返回给消费者
  • 消费者仅需收到消息后,再在客户端自己这边执行一个用户自定义的回调就行了!
import lombok.Getter;
import lombok.Setter;import java.io.Serializable;@Getter
@Setter
public class BasicConsumeArguments extends BasicArguments implements Serializable {private String consumerTag;private String queueName;private boolean autoAck;
//    这个类对应的 basicConsume 方法中,还有一个参数,是回调函数(如何来有效处理消息)
//    这个回调函数,是不能通过网络传输的
//    站在 broker server 这边,针对消息的处理问题,其实是统一的(把消息返回给客户端)
//    客户端这边收到消息之后,再在客户端自己这边执行一个用户自定义的回调就行了
//    此时客户端就不需要把自身的回调告诉服务器了!
//    这个类就不需要 consumer 成员了
}

特列二:

  • type = 0xc,即 服务器给客户端推送消息(被订阅的消息),该类型一定是一个响应!

  • 如上图所示的蓝色部分
  • 此处我们定义一个 SubScribeReturns 类用于表示在消费者订阅队列之后,服务器给消费推送消息的响应参数
  • 此处仍需继承一下 代表响应公共参数的 BasicReturns 类
import com.example.demo.mqserver.core.BasicProperties;
import lombok.Getter;
import lombok.Setter;import java.io.Serializable;@Getter
@Setter
public class SubScribeReturns extends BasicReturns implements Serializable {private String consumerTag;private BasicProperties basicProperties;private byte[] body;
}

注意:

  • SubScribeReturns 类虽然继承了 BasicReturns 类
  • 但是在返回时,无需填写 BasicReturns 类中的成员变量 rid
  • 因为该响应无相对应的请求,故该响应无 rid,即将 rid 设为空字符串即可

小结:

  • 上述内容属于服务器程序的关键环节,自定义应用层协议

 实现 Broker Server 类

属性 与 构造

/*
* 这个 BrokerServer 就是咱们 消息队列 本体服务器
* 本质上就是一个 TCP 的服务器
* */
public class BrokerServer {private ServerSocket serverSocket = null;//    当前考虑一个 BrokerServer 上只有一个 虚拟主机private VirtualHost virtualHost = new VirtualHost("default");
//    使用这个 哈希表 表示当前的所有会话(也就是说有哪些客户端正在和咱们的服务器进行通信)
//    此处的 key 是 channelId,value 为对应的 Socket 对象private ConcurrentHashMap<String, Socket> sessions = new ConcurrentHashMap<String,Socket>();
//    引入线程池,来处理多个客户端的请求private ExecutorService executorService = null;
//    引入一个 Boolean 变量控制服务器是否继续运行private volatile boolean runnable = true;public BrokerServer(int port) throws IOException {serverSocket = new ServerSocket(port);}
}

启动 Broker Server

public void start() throws IOException {System.out.println("[BrokerServer] 启动!");executorService = Executors.newCachedThreadPool();try {while (runnable) {Socket clientSocket = serverSocket.accept();
//            把处理连接的逻辑丢给这个线程池executorService.submit(() ->{processConnection(clientSocket);});}}catch (SocketException e){System.out.println("[BrokerServer] 服务器停止运行!");}}

停止 Broker Server

//    一般来说停止服务器,就是直接 kill 掉对应进程就行了
//    此处还是搞一个单独的停止方法,主要是用于后续的单元测试public void stop() throws IOException {runnable = false;
//        把线程池中的任务都放弃了,让线程都销毁executorService.shutdownNow();serverSocket.close();}

处理客户端连接

//    通过这个方法来处理一个客户端的连接
//    在这一个连接中,可能会涉及到多个请求和响应private void processConnection(Socket clientSocket){try (InputStream inputStream = clientSocket.getInputStream();OutputStream outputStream = clientSocket.getOutputStream()){
//            这里需要按照特定格式来读取并解析,此时就需要用到 DataInputStream 和 DataOutputStreamtry (DataInputStream dataInputStream = new DataInputStream(inputStream);DataOutputStream dataOutputStream = new DataOutputStream(outputStream)){while (true) {
//                    1、读取请求并解析Request request = readRequest(dataInputStream);
//                    2、根据请求计算响应Response response = process(request,clientSocket);
//                    3、把响应写回给客户端writeResponse(dataOutputStream,response);}}} catch (EOFException | SocketException e) {
//                对于这个代码,DataInputStream 如果读到 EOF,就会抛出一个 EOFException 异常
//                需要借助这个异常来结束循环System.out.println("[BrokerServer] connection 关闭!客户端的地址:" + clientSocket.getInetAddress().toString()+ ":" + clientSocket.getPort());} catch (IOException | ClassNotFoundException | MqException e) {System.out.println("[BrokerServer] connection 出现异常!");e.printStackTrace();}finally {try {
//            当连接处理完了,就需要记得关闭 socketclientSocket.close();
//            一个 TCP 连接中,可能包含多个 channel 需要把当前这个 socket 对应的所有 channel 也顺便清理掉clearClosedSession(clientSocket);}catch (IOException e) {e.printStackTrace();}}}

读取请求 与 写回响应

    private Request readRequest(DataInputStream dataInputStream) throws IOException {Request request = new Request();request.setType(dataInputStream.readInt());request.setLength(dataInputStream.readInt());byte[] payload = new byte[request.getLength()];int n = dataInputStream.read(payload);if(n != request.getLength()) {throw new IOException("读取请求格式出错!");}request.setPayload(payload);return request;}private void writeResponse(DataOutputStream dataOutputStream, Response response) throws IOException {dataOutputStream.writeInt(response.getType());dataOutputStream.writeInt(response.getLength());dataOutputStream.write(response.getPayload());
//        这个刷新缓冲区也是重要的操作,保证当前写的这些数据能够快速进入到网卡里,而不至于在内存中呆着dataOutputStream.flush();}

根据请求计算响应

  • 根据不同的 type 类型,来远程调用 VirtualHost 中不同的核心 API

具体代码编写:

private Response process(Request request, Socket clientSocket) throws IOException, ClassNotFoundException, MqException {
//        1、把 request 中的 payload 做一个初步的解析BasicArguments basicArguments = (BasicArguments) BinaryTool.fromBytes(request.getPayload());System.out.println("[Request] rid = " + basicArguments.getRid() + ", channelId = " + basicArguments.getChannelId()+ ", type = " + request.getType() + ", length = " + request.getLength());
//        2、根据 type 的值,来进一步区分接下来这次请求要干啥boolean ok = true;if(request.getType() == 0x1) {
//            创建 channelsessions.put(basicArguments.getChannelId(), clientSocket);System.out.println("[BrokerServer] 创建 channel 完成! channelId = " + basicArguments.getChannelId());}else if(request.getType() == 0x2) {
//            销毁 channelsessions.remove(basicArguments.getChannelId());System.out.println("[BrokerServer] 销毁 channel 完成! channelId = " + basicArguments.getChannelId());} else if(request.getType() == 0x3) {
//            创建交换机,此时 payload 就是 ExchangeDeclareArguments 对象了ExchangeDeclareArguments arguments = (ExchangeDeclareArguments) basicArguments;ok = virtualHost.exchangeDeclare(arguments.getExchangeName(),arguments.getExchangeType(),arguments.isDurable(),arguments.isAutoDelete(),arguments.getArguments());} else if(request.getType() == 0x4) {
//            删除交换机,此时 payload 就是 ExchangeDeleteArguments 对象了ExchangeDeleteArguments arguments = (ExchangeDeleteArguments) basicArguments;ok = virtualHost.exchangeDelete(arguments.getExchangeName());} else if(request.getType() == 0x5) {
//            创建队列,此时 payload 就是 QueueDeclareArguments 对象了QueueDeclareArguments arguments = (QueueDeclareArguments) basicArguments;ok = virtualHost.queueDeclare(arguments.getQueueName(),arguments.isDurable(),arguments.isExclusive(),arguments.isAutoDelete(),arguments.getArguments());} else if(request.getType() == 0x6){
//            销毁队列,此时 payload 就是 QueueDeleteArguments 对象了QueueDeleteArguments arguments = (QueueDeleteArguments) basicArguments;ok = virtualHost.queueDelete(arguments.getQueueName());} else if(request.getType() == 0x7){
//            创建绑定,此时 payload 就是 QueueBindArguments 对象了QueueBindArguments arguments = (QueueBindArguments) basicArguments;ok = virtualHost.queueBind(arguments.getQueueName(), arguments.getExchangeName(), arguments.getBindingKey());} else if(request.getType() == 0x8){
//            删除绑定,此时 payload 就是 QueueUnbindArguments 对象了QueueUnbindArguments arguments = (QueueUnbindArguments) basicArguments;ok = virtualHost.queueUnbind(arguments.getQueueName(),arguments.getExchangeName());} else if(request.getType() == 0x9){
//            发送消息,此时 payload 就是 BasicPublishArguments 对象了BasicPublishArguments arguments = (BasicPublishArguments) basicArguments;ok = virtualHost.basicPublish(arguments.getExchangeName(),arguments.getRoutingKey(),arguments.getBasicProperties(),arguments.getBody());} else if(request.getType() == 0xa){
//            订阅消息,此时 payload 就是 BasicConsumeArguments 对象了BasicConsumeArguments arguments = (BasicConsumeArguments) basicArguments;ok = virtualHost.basicConsume(arguments.getConsumerTag(), arguments.getQueueName(), arguments.isAutoAck(), new Consumer() {
//                    这个回调函数要做的工作,就是把服务器收到的消息可以直接推送回对应的消费者客户端@Overridepublic void handleDelivery(String consumerTag, BasicProperties basicProperties, byte[] body) throws MqException, IOException {
//                    先知道当前这个收到的消息,要发给哪个客户端,此处 consumerTag 其实是 channelId
//                    根据 channelId 去 sessions 中查询,就可以得到对应的 socket 对象了,从而可以往里面发送数据了
//                    1、根据 channelId 找到 socket 对象Socket clientSocket = sessions.get(consumerTag);if(clientSocket == null || clientSocket.isClosed()) {throw new MqException("[BrokerServer] 订阅消息的客户端已经关闭!");}
//                    2、构造响应数据SubScribeReturns subScribeReturns = new SubScribeReturns();subScribeReturns.setChannelId(consumerTag);subScribeReturns.setRid(""); // 由于这里只有响应,没有请求,不需要去对应 rid 暂时不需要subScribeReturns.setOk(true);subScribeReturns.setConsumerTag(consumerTag);subScribeReturns.setBody(body);subScribeReturns.setBasicProperties(basicProperties);byte[] paylaod = BinaryTool.toBytes(subScribeReturns);Response response = new Response();
//                    0xc 表示服务器给消费者客户端推送的消息数据response.setType(0xc);
//                    response 的 payload 就是一个 SubScribeReturnsresponse.setLength(paylaod.length);response.setPayload(paylaod);
//                    3、把数据写回给客户端
//                    注意!此处的 dataOutputStream 这个对象不能 close!
//                    如果把 dataOutputStream 关闭,就会直接把 clientSocket 里的 outputStream 也给关了
//                    此时就无法继续往 socket 中写入后续数据了!DataOutputStream dataOutputStream = new DataOutputStream(clientSocket.getOutputStream());writeResponse(dataOutputStream,response);}});} else if(request.getType() == 0xb){
//            确认消息,此时 payload 就是 BasicAckArguments 对象了BasicAckArguments arguments = (BasicAckArguments) basicArguments;ok = virtualHost.basicAck(arguments.getQueueName(), arguments.getMessageId());}else {
//            当前的 type 是非法的throw new MqException("[BrokerServer] 未知的 type!type = " + request.getType());}
//        3、构造响应BasicReturns basicReturns = new BasicReturns();basicReturns.setChannelId(basicArguments.getChannelId());basicReturns.setRid(basicArguments.getRid());basicReturns.setOk(ok);byte[] payload = BinaryTool.toBytes(basicReturns);Response response = new Response();response.setType(request.getType());response.setLength(payload.length);response.setPayload(payload);System.out.println("[Response] rid = " + basicReturns.getRid() + ",channelId = " + basicReturns.getChannelId()+ ", type = " + response.getType() + ",length = " + response.getLength());return response;}

注意点一:

  • 当前请求中的 payload 里面放的内容 是根据 type 的类型来走的
  • 比如 type 是 0x3,payload 就是 ExchangeDeclareArguments
  • 比如 type 是 0x4,payload 就是 ExchangeDeleteArguments
  • ...

注意点二:

  • 此处设定的不同的方法的参数,虽然都有不同的类
  • 但是它们均继承自同一个 BasicArguments 类
  • 因此先将 payload 转成 BasicArguments

清除 channel 

  • 清理 sessions 这个 哈希表 中的 session 信息

具体代码编写:

     private void clearClosedSession(Socket clientSocket) {
//         这里要做的事情,主要就是遍历上述 session hash 表,把该关闭的 socket 对应的键值对,统统删掉List<String> toDeleteChannelId = new ArrayList<>();for(Map.Entry<String,Socket> entry : sessions.entrySet()) {if(entry.getValue() == clientSocket) {
//                不能在这里直接删除
//                这属于集合类的一个大忌!!一边遍历,一边删除!
//                session.remove(entry.getKey());toDeleteChannelId.add(entry.getKey());}}for (String channelId : toDeleteChannelId) {sessions.remove(channelId);}System.out.println("[BrokerServer] 清理 session 完成!被清理的 channelId = " + toDeleteChannelId);}

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Onlyfans 地址错误、无法支付、年龄验证等问题解决方案!!!

很多客户在 Onlyfans 绑卡时&#xff0c;出现了地址错误&#xff0c;年龄验证&#xff0c;无法支付等各种问题。 出现这个问题的原因&#xff1a; 一是用国内邮箱注册了&#xff0c; 二是 ip 有问题&#xff0c;会导致出现年龄验证&#xff0c;或无法支付 Onlyfans 等问题。…

OpenHarmony 分布式开发实战——线上菜单

简介 分布式菜单demo 模拟的是多人聚餐点菜的场景&#xff0c;不需要扫码关注公众号等一系列操作&#xff0c;通过分布式数据库可以方便每个人可及时查看到订单详情&#xff0c;数量&#xff0c;总额等&#xff1b;效果如下 demo效果 工程目录 完整的项目结构目录如下 ├─…

linux系统---httpd

目录 Internet的起源 一、http协议——超文本传输协议 1.http相关概念 二、HTTP请求访问的完整过程 1、 建立连接 2、 接收请求 3、 处理请求 常用请求Method: GET、POST、HEAD、PUT、DELETE、TRACE、OPTIONS 3.1 常见的HTTP方法 3.2 GET和POST比较 4、访问资源 …

【postgresql】数据表id自增与python sqlachemy结合实例

需求&#xff1a; postgresql实现一个建表语句&#xff0c;表名&#xff1a;student,字段id,name,age&#xff0c; 要求&#xff1a;每次添加一个数据id会自动增加1 在PostgreSQL中&#xff0c;您可以使用SERIAL或BIGSERIAL数据类型来自动生成主键ID。以下是一个创建名为stude…

MongoDB之客户端工具与核心概念及基本类型篇

MongoDB之客户端工具与核心概念及基本类型篇 文章目录 MongoDB之客户端工具与核心概念及基本类型篇1. MongoDB是什么?1. 关于MongoDB2. 相关客户端工具1. MongoDB Compass2. Studio 3T3. Navicat for MongoDB4. NoSQL Manager for MongoDB Professional 2.MongoDB相关概念2.1 …

4.测试教程 - 用例篇

文章目录 1.测试用例的基本要素2.测试用例的给我们带来的好处3.测试用例的设计方法3.1基于需求进行测试用例的设计3.1.1功能需求测试分析3.1.2非功能需求测试分析 3.2具体的设计方法3.2.1等价类3.2.2边界值3.2.3错误猜测法3.2.4判定表3.2.5场景设计法3.2.6因果图3.2.7因果图的需…

Python 鼠标模拟

鼠标模拟即&#xff1a;通过python 进行模拟鼠标操作 引入类库 示例如下&#xff1a; import win32api import win32con import time 设置鼠标位置 设置鼠标位置为窗口中的回收站。 示例如下&#xff1a; # 设置鼠标的位置 win32api.SetCursorPos([30, 40]) 双击图标 设置…

springboot+vue前后端分离适配cas认证的跨域问题

0. cas服务搭建参考:CAS 5.3服务器搭建_cas-overlay-CSDN博客 1. 参照springsecurity适配cas的方式, 一直失败, 无奈关闭springssecurity认证 2. 后端服务适配cas: 参考前后端分离项目(springbootvue)接入单点登录cas_前后端分离做cas单点登录-CSDN博客 1) 引入maven依赖 …

如何系统地自学 Python

设定学习目标 确定自己学习 Python 的目的和用途&#xff0c;这一步很重要&#xff0c;比如是为了编写脚本、开发网站、进行数据分析等。设定清晰的学习目标&#xff0c;把目标拆分为一个个阶段的小目标&#xff0c;通过完成一个个小目标&#xff0c;得到正反馈&#xff0c;激…

SD-WAN解决企业组网中网络卡顿问题

网络卡顿已成为企业组网中一大难题&#xff0c;特别是随着办公应用系统的内网服务器或云端部署&#xff0c;员工对网络的依赖日益增加。面对网络卡顿问题&#xff0c;我们不得不深入思考如何提升工作效率并改善员工体验。本文将深入探讨企业组网中的网络问题&#xff0c;并介绍…

DeepMind基础世界模型Genie:一张草图即为一个世界,通用AI智能体要来了?

一张草图即为一个世界&#xff01;Google DeepMind 推出了首个以无监督方式从未经标注的互联网视频中训练而来的生成交互环境模型——Genie。该模型可以通过文本、合成图像、照片甚至草图来生成无数种可玩&#xff08;动作可控&#xff09;的虚拟世界。 据介绍&#xff0c;Geni…

使用管道和system V进行进程间通信

进程通信的目的 数据传输&#xff1a;一个进程需要将它的数据发送给另一个进程资源共享&#xff1a;多个进程之间共享同样的资源。通知事件&#xff1a;一个进程需要向另一个或一组进程发送消息&#xff0c;通知它&#xff08;它们&#xff09;发生了某种事件&#xff08;如进程…

投资生涯的核心密码:构建交易逻辑体系

首先&#xff0c;我们需要明确一点&#xff0c;交易中究竟有没有确定性&#xff1f; 确定性是指在某一种形式、或有若干条件时&#xff0c;价格必然会上涨或下跌&#xff0c;也可以决定上涨或下跌的程度。 我认为&#xff0c;没有。迄今为止还没有一个理论能发现即使确定的东西…

python图像处理初步

文章目录 处理流程灰度分布图 处理流程 在Python中&#xff0c;通过【plt】和【numpy】可以实现图像处理的最简单的流程&#xff0c;即读取图片->处理图片->显示结果->保存结果。 import matplotlib.pyplot as plt import numpy as nppath lena.jpg img plt.imrea…

春节医美热,爱美客、昊海生科谁更赚钱?

在颜值经济赛道上&#xff0c;医美项目逐渐成为消费主流。随着春节假期的到来&#xff0c;医美消费又将迎来高峰期。 “医美三剑客”中&#xff0c;爱美客(300896.SZ)、昊海生科(688366.SH)近日相继公布了2023年的业绩报告&#xff1a;2023年&#xff0c;爱美客预计实现净利润…

白敬亭风波后现身,心情低落进新剧组,父母暖心陪伴。

♥ 为方便您进行讨论和分享&#xff0c;同时也为能带给您不一样的参与感。请您在阅读本文之前&#xff0c;点击一下“关注”&#xff0c;非常感谢您的支持&#xff01; 文 |猴哥聊娱乐 编 辑|徐 婷 校 对|侯欢庭 白敬亭春晚首秀引热议&#xff0c;口碑因“春山学”风波陷两极…

掌握Docker:让你的应用轻松部署和管理

文章目录 一、引言&#xff08;为什么要学习docker&#xff1f;&#xff09;1.1 环境不一致1.2 隔离性1.3 弹性伸缩1.4 学习成本 二、Docker介绍2.1 Docker的由来2.2 什么是Docker2.3 为什么要用Docker2.3.1 虚拟机2.3.2 Linux容器 2.4 Docker与传统虚拟机的区别2.5 Docker的思…

微信小程序(四十五)登入界面-简易版

注释很详细&#xff0c;直接上代码 上一篇 此文使用了vant组件库&#xff0c;没有安装配置的可以参考此篇vant组件的安装与配置 新增内容&#xff1a; 1.基础组件的组合 2.验证码倒计时的逻辑处理 源码&#xff1a; app.json {"usingComponents": {"van-field…

打印水仙花数---c语言刷题

欢迎关注个人主页&#xff1a;逸狼 创造不易&#xff0c;可以点点赞吗~ 如有错误&#xff0c;欢迎指出~ 题述 求出0&#xff5e;100000之间的所有“水仙花数”并输出。 “水仙花数”是指一个n位数&#xff0c;其各位数字的n次方之和确好等于该数本身&#xff0c;如:153&#…

C++数据库连接池

功能实现设计 &#xff1a; ConnectionPool.cpp 和 ConnectionPool.h &#xff1a;连接池代码实现 Connection.cpp 和 Connection.h &#xff1a;数据库操作代码、增删改查代码实现 连接池主要包含了以下功能点 &#xff1a; 1.连接池只需要一个实例&#xff0c;所以 Connec…