当前位置: 首页 > news >正文

处理任务“无需等待”:集成RabbitMQ实现异步通信与系统解耦

在前几篇文章中,我们构建的Web应用遵循了一个常见的同步处理模式:用户发出HTTP请求 -> Controller接收 -> Service处理(可能涉及数据库操作、调用其他内部方法)-> Controller返回HTTP响应。这个流程简单直接,但在某些场景下会遇到瓶颈:

  • 用户体验不佳: 如果Service层需要执行一些耗时操作(比如发送邮件/短信、生成复杂报表、调用外部慢API、进行大量计算),用户就必须一直等待直到所有操作完成,才能收到响应。这会导致页面卡顿,用户体验直线下降。

  • 系统耦合度高: 如果一个服务(比如订单服务)需要通知另一个服务(比如库存服务和通知服务),直接通过RPC或HTTP调用,会导致服务之间紧密耦合。如果被调用服务暂时不可用或处理缓慢,会直接影响调用方(订单服务)的性能和可用性。

  • 流量洪峰处理能力差: 如果短时间内涌入大量请求(如秒杀活动),所有请求都直接冲击后端服务和数据库,很容易导致系统过载甚至崩溃。

如何解决这些问题?引入异步处理系统解耦是关键,而消息队列 (MQ) 正是实现这两者的利器。

想象一下去银行办理业务,如果每个柜员(服务)都必须等上一个客户完全办完所有流程(包括需要后台审批的耗时环节)才接待下一个,效率会非常低。而引入叫号系统(消息队列)后,你取号(发送消息),然后可以坐下等待(主流程结束),当柜员空闲时,会叫到你的号(消费消息)来处理你的业务。这大大提高了整体效率和用户体验。

读完本文,你将学会:

  • 理解为什么需要消息队列以及它的核心优势。

  • 了解RabbitMQ的基本概念(生产者、消费者、队列、交换机)。

  • 掌握如何使用spring-boot-starter-amqp轻松集成RabbitMQ。

  • 通过RabbitTemplate发送消息(简单文本和Java对象)。

  • 使用@RabbitListener注解异步接收并处理消息。

  • (可选)了解如何通过Java配置声明队列、交换机和绑定。

准备好让你的应用学会“异步分身术”,提升响应速度和系统韧性了吗?

一、为什么需要消息队列?核心优势解析

消息队列是一种提供异步通信机制的中间件。它允许不同的应用程序或服务通过发送和接收消息来进行通信,而无需直接相互连接。

核心优势:

  1. 异步处理 (Asynchronous Processing):

    • 场景: 用户注册后需要发送欢迎邮件。

    • 同步方式: 保存用户信息 -> 调用邮件发送接口 -> 等待邮件发送成功 -> 返回注册成功响应给用户。如果邮件接口慢,用户注册就会很慢。

    • 异步方式: 保存用户信息 -> 发送一个“发送欢迎邮件”的消息到MQ -> 立即返回注册成功响应给用户。后台有一个独立的邮件服务会从MQ消费这个消息并执行发送操作。

    • 效果: 用户注册响应速度大大提升。

  2. 应用解耦 (Decoupling):

    • 场景: 订单创建成功后,需要通知库存服务扣减库存、通知物流服务准备发货、通知积分服务增加积分。

    • 紧耦合方式: 订单服务依次调用库存、物流、积分服务的接口。任何一个下游服务接口变更或不可用,都会影响订单服务。

    • MQ方式: 订单服务只需要发送一个“订单已创建”的消息(包含订单信息)到MQ。库存、物流、积分服务各自订阅这个消息,独立进行处理。

    • 效果: 订单服务不再强依赖下游服务,下游服务增减或变更对订单服务透明。系统更灵活、易于扩展。

  3. 削峰填谷 (Traffic Shaping / Load Leveling):

    • 场景: 秒杀活动开始瞬间,大量下单请求涌入。

    • 直接处理: 所有请求直接打到订单服务和数据库,很容易超出处理能力导致系统崩溃。

    • MQ方式: 前端应用或网关快速接收请求,将下单请求转化为消息放入MQ。后端的订单处理服务按照自己的节奏(比如每秒处理100个)从MQ中拉取消息进行处理。

    • 效果: MQ作为缓冲区,平滑了流量洪峰,保护了后端系统不被打垮,保证了系统的稳定性。

二、初识RabbitMQ:核心概念速览

RabbitMQ是一个实现了AMQP(高级消息队列协议)的、流行的、开源的消息代理(Message Broker)。理解以下几个核心概念对于使用它至关重要:

  • Producer (生产者): 发送消息的应用程序。

  • Consumer (消费者): 接收并处理消息的应用程序。

  • Broker (代理): RabbitMQ服务器本身,负责接收、存储和路由消息。

  • Queue (队列): 消息存储的缓冲区,位于Broker内部。消息从生产者发出后,最终被路由到队列中等待消费者处理。多个消费者可以监听同一个队列(但一条消息通常只会被一个消费者处理 - P2P模式)。

  • Exchange (交换机): 接收来自生产者的消息,并根据路由规则 (Routing Key) 将消息路由到一个或多个队列。生产者实际上是将消息发送到Exchange。Exchange有几种类型,决定了路由逻辑:

    • Direct Exchange: 根据Routing Key精确匹配,将消息路由到Binding Key与之完全相同的队列。

    • Fanout Exchange: 忽略Routing Key,将消息广播到所有绑定到它的队列。

    • Topic Exchange: 根据Routing Key进行模式匹配(使用 * 匹配一个单词,# 匹配零个或多个单词),将消息路由到匹配模式的队列。

    • Headers Exchange: 根据消息头中的属性进行匹配(不常用)。

  • Binding (绑定): 定义Exchange和Queue之间的连接关系。对于Direct和Topic Exchange,Binding通常还包含一个Binding Key,用于匹配消息的Routing Key。

  • Message (消息): 生产者和消费者之间传递的数据。通常包含两部分:Payload (消息体) 和 Headers (消息头,可选的属性)

简化流程 (以Direct Exchange为例):
Producer -(消息 + Routing Key A)-> Exchange -(Binding Key A)-> Queue A <- Consumer A
Producer -(消息 + Routing Key B)-> Exchange -(Binding Key B)-> Queue B <- Consumer B

三、Spring Boot集成:spring-boot-starter-amqp

Spring Boot通过spring-boot-starter-amqp模块极大地简化了与RabbitMQ(以及其他AMQP兼容的Broker)的集成。

1. 添加依赖 (Maven):

<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-amqp</artifactId>
</dependency>

2. 配置连接信息 (application.yml):

spring:rabbitmq:host: localhost       # RabbitMQ服务器地址 (默认localhost)port: 5672            # RabbitMQ端口 (默认5672)username: guest       # 用户名 (默认guest)password: guest       # 密码 (默认guest)# virtual-host: /     # 虚拟主机 (默认/)# publisher-confirm-type: correlated # (可选) 开启发送方确认模式# publisher-returns: true            # (可选) 开启发送失败退回模式# template:#   mandatory: true                # (可选) 配合publisher-returns, 确保消息至少路由到一个队列

注意: 生产环境中,用户名和密码应使用上一篇文章介绍的配置管理方式(如环境变量、外部文件)注入,而非硬编码。

配置完成后,Spring Boot会自动配置好连接工厂 (ConnectionFactory)、管理模板 (RabbitAdmin) 以及发送消息的核心工具 RabbitTemplate。

四、发送消息 (Producer): RabbitTemplate

RabbitTemplate是Spring AMQP提供的用于发送消息的核心类。

示例:用户注册后异步发送欢迎邮件通知

  1. 修改UserService (注入RabbitTemplate):

    package com.example.service;import com.example.model.User;
    import com.example.repository.UserRepository;
    import org.springframework.amqp.rabbit.core.RabbitTemplate; // 导入
    import org.springframework.beans.factory.annotation.Autowired;
    import org.springframework.stereotype.Service;
    import org.springframework.transaction.annotation.Transactional;@Service
    public class UserService {private final UserRepository userRepository;private final RabbitTemplate rabbitTemplate; // 注入RabbitTemplate// 定义队列名称 (最好定义为常量或配置)public static final String WELCOME_EMAIL_QUEUE = "q.user.welcome.email";// 定义交换机名称 (使用默认Direct交换机时为空字符串, RoutingKey就是QueueName)public static final String DEFAULT_EXCHANGE = ""; // 空字符串代表默认交换机@Autowiredpublic UserService(UserRepository userRepository, RabbitTemplate rabbitTemplate) {this.userRepository = userRepository;this.rabbitTemplate = rabbitTemplate;}@Transactionalpublic User createUser(String name, String email, Integer age) {User newUser = new User(name, email, age);User savedUser = userRepository.save(newUser);System.out.println("Saved user to DB: " + savedUser);// --- 异步发送消息 ---try {// 发送消息到指定队列 (使用默认交换机, routingKey就是队列名)// convertAndSend 会自动将 User 对象序列化 (通常为JSON)rabbitTemplate.convertAndSend(DEFAULT_EXCHANGE, WELCOME_EMAIL_QUEUE, savedUser);System.out.println("Sent welcome email task message for user: " + savedUser.getEmail());// 你也可以只发送必要的ID或信息, 而不是整个对象// rabbitTemplate.convertAndSend(WELCOME_EMAIL_QUEUE, savedUser.getId());} catch (Exception e) {// 考虑: 消息发送失败的处理策略 (记录日志, 补偿任务等)System.err.println("Failed to send welcome email task message: " + e.getMessage());// 注意: 这里不应影响主事务的回滚 (如果需要的话)}return savedUser;}// ... 其他方法 ...
    }

    convertAndSend(String exchange, String routingKey, Object message) 是最常用的发送方法。它会自动处理对象到消息的转换(默认使用Jackson2JsonMessageConverter转为JSON)。

五、接收消息 (Consumer): @RabbitListener

通过@RabbitListener注解,可以非常方便地创建消息消费者。

示例:创建邮件服务消费者来处理欢迎邮件任务

  1. 创建EmailConsumer组件:

    package com.example.consumer;import com.example.model.User; // 需要能访问User类
    import org.springframework.amqp.rabbit.annotation.RabbitListener; // 导入
    import org.springframework.stereotype.Component;@Component
    public class EmailConsumer {// 使用 @RabbitListener 注解监听指定队列// Spring AMQP 会自动创建队列 (如果不存在且配置允许)@RabbitListener(queues = UserService.WELCOME_EMAIL_QUEUE)public void handleWelcomeEmail(User user) { // 参数类型与发送时一致 (或Object/Message)System.out.println("Received welcome email task for user: " + user);try {// --- 模拟发送邮件的耗时操作 ---System.out.println("Simulating sending welcome email to " + user.getEmail() + "...");Thread.sleep(2000); // 模拟耗时2秒System.out.println("Welcome email sent successfully to " + user.getEmail());// 如果处理成功, Spring AMQP 会自动发送 ACK (消息确认) 给RabbitMQ// RabbitMQ 确认后会从队列中删除该消息} catch (InterruptedException e) {Thread.currentThread().interrupt();System.err.println("Email sending task interrupted for user: " + user.getEmail());// 抛出异常会导致消息处理失败throw new RuntimeException("Email sending interrupted", e);} catch (Exception e) {// 其他异常也可能导致处理失败System.err.println("Error sending welcome email to " + user.getEmail() + ": " + e.getMessage());// 如果方法抛出异常, Spring AMQP 默认会拒绝消息 (NACK)// 根据配置, 消息可能会被重新入队 (可能导致死循环!) 或进入死信队列 (推荐)throw e; // 重新抛出, 让Spring AMQP知道处理失败}}// 可以监听同一个队列的多个实例 (用于提高并发处理能力)// @RabbitListener(queues = UserService.WELCOME_EMAIL_QUEUE)// public void handleWelcomeEmailInstance2(User user) { ... }// 监听其他队列// @RabbitListener(queues = "another.queue")// public void handleAnotherTask(String messagePayload) { ... }
    }

    • @RabbitListener(queues = "..."): 指定要监听的队列名称。

    • 方法参数可以直接是消息体反序列化后的对象类型(如User)。Spring AMQP会自动完成转换。也可以是org.springframework.amqp.core.Message获取完整消息,或com.rabbitmq.client.Channel进行手动ACK等高级操作。

    • 消息确认 (Acknowledgement, ACK): 默认情况下,如果@RabbitListener方法成功执行完毕(没有抛出异常),Spring AMQP会自动向RabbitMQ发送ACK,告知消息已被成功处理,可以从队列中删除了。如果方法抛出异常,则会发送NACK(或Reject),消息可能会被重新投递或进入死信队列(需要额外配置)。这是保证消息不丢失的关键机制。

六、最佳实践:声明式定义基础设施

虽然RabbitTemplate和@RabbitListener在某些配置下可以自动创建队列,但在生产环境中,推荐显式地声明所需的队列、交换机和绑定。这能确保基础设施的存在,避免因自动创建的不可靠性导致问题,并且使配置更清晰。

可以通过在@Configuration类中定义Queue, Exchange, Binding类型的Bean来实现:

package com.example.config;import org.springframework.amqp.core.*; // 导入核心类
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;@Configuration
public class RabbitMQConfig {// --- 声明欢迎邮件队列 ---@Beanpublic Queue welcomeEmailQueue() {// durable(true) 持久化队列 (RabbitMQ重启后依然存在)return new Queue(UserService.WELCOME_EMAIL_QUEUE, true);}// --- (可选) 如果不使用默认交换机, 可以声明一个交换机 ---// 例如, 声明一个 Direct Exchange// @Bean// public DirectExchange userEventsExchange() {//     return new DirectExchange("x.user.events", true, false);// }// --- (可选) 声明绑定关系 ---// 将欢迎邮件队列绑定到默认交换机 (RoutingKey就是队列名)@Beanpublic Binding welcomeEmailBinding(Queue welcomeEmailQueue) {// 目标 (队列), 类型 (队列), 交换机 (默认), RoutingKey, 参数return BindingBuilder.bind(welcomeEmailQueue).to(DirectExchange.DEFAULT).withQueueName();}// 如果使用了自定义交换机:// @Bean// public Binding welcomeEmailBindingToUserExchange(Queue welcomeEmailQueue, DirectExchange userEventsExchange) {//    return BindingBuilder.bind(welcomeEmailQueue).to(userEventsExchange).with(UserService.WELCOME_EMAIL_QUEUE); // 使用队列名作为RoutingKey// }// ---- 可以声明其他队列、交换机和绑定 ----// @Bean public Queue orderCreatedQueue() { ... }// @Bean public FanoutExchange notificationExchange() { ... }// @Bean public Binding orderNotificationBinding(Queue orderCreatedQueue, FanoutExchange notificationExchange) { ... }
}

Spring AMQP启动时会检查这些Bean,如果对应的队列、交换机或绑定在RabbitMQ中不存在,RabbitAdmin会自动创建它们。

七、何时使用消息队列?

  • 需要将耗时操作从主流程中剥离,提高用户响应速度时(如邮件发送、报表生成)。

  • 需要解耦不同服务或模块之间的依赖关系时(如订单与库存、物流、积分)。

  • 需要缓冲突发流量,保护后端系统时(如秒杀、批量数据导入)。

  • 构建事件驱动架构时。

八、总结:开启异步与解耦的新篇章

消息队列(如RabbitMQ)是构建健壮、可扩展的现代分布式系统的重要工具。通过引入异步处理和应用解耦,它可以显著提升用户体验、系统灵活性和稳定性。Spring Boot AMQP (spring-boot-starter-amqp) 提供了与RabbitMQ无缝集成的能力,通过RabbitTemplate发送消息和@RabbitListener消费消息,使得在Spring应用中使用MQ变得异常简单。

掌握消息队列的集成与使用,将为你的应用程序架构设计打开新的思路,助你构建更加高效、可靠的系统。

相关文章:

处理任务“无需等待”:集成RabbitMQ实现异步通信与系统解耦

在前几篇文章中&#xff0c;我们构建的Web应用遵循了一个常见的同步处理模式&#xff1a;用户发出HTTP请求 -> Controller接收 -> Service处理&#xff08;可能涉及数据库操作、调用其他内部方法&#xff09;-> Controller返回HTTP响应。这个流程简单直接&#xff0c;…...

ASP.NET Core 主机模型详解:Host、WebHost与WebApplication的对比与实践【代码之美】

&#x1f380;&#x1f380;&#x1f380;代码之美系列目录&#x1f380;&#x1f380;&#x1f380; 一、C# 命名规则规范 二、C# 代码约定规范 三、C# 参数类型约束 四、浅析 B/S 应用程序体系结构原则 五、浅析 C# Async 和 Await 六、浅析 ASP.NET Core SignalR 双工通信 …...

编译型语言、解释型语言与混合型语言:原理、区别与应用场景详解

编译型语言、解释型语言与混合型语言&#xff1a;原理、区别与应用场景详解 文章目录 编译型语言、解释型语言与混合型语言&#xff1a;原理、区别与应用场景详解引言一、编译型语言1.1 工作原理1.2 典型的编译型语言1.3 优点1.4 缺点 二、解释型语言2.1 工作原理2.2 典型的解释…...

AI工程pytorch小白TorchServe部署模型服务

注意&#xff1a;该博客仅是介绍整体流程和环境部署&#xff0c;不能直接拿来即用(避免公司代码外泄)请理解。并且当前流程是公司notebook运行&本机windows&#xff0c;后面可以使用docker 部署镜像到k8s&#xff0c;敬请期待~ 前提提要&#xff1a;工程要放弃采购的AI平台…...

Ubuntu 一站式部署 RabbitMQ 4 并“彻底”迁移数据目录的终极实践

1 安装前准备 sudo apt update -y sudo apt install -y curl gnupg apt-transport-https lsb-release jq若计划将数据放到新磁盘&#xff08;如 /dev/nvme0n1p1&#xff09;&#xff1a; sudo mkfs.xfs /dev/nvme0n1p1 sudo mkdir /data echo /dev/nvme0n1p1 /data xfs defau…...

华为手机怎么进行音频降噪?音频降噪技巧分享:提升听觉体验

在当今数字化时代&#xff0c;音频质量对于提升用户体验至关重要&#xff0c;无论是在通话、视频录制还是音频文件播放中&#xff0c;清晰的音频都能带来更佳的听觉享受。 而华为手机凭借其强大的音频处理技术&#xff0c;为用户提供了多种音频降噪功能&#xff0c;帮助用户在…...

拥抱健康生活,解锁养生之道

在生活节奏日益加快的当下&#xff0c;健康养生已成为人们关注的焦点。科学的养生方法&#xff0c;能帮助我们增强体质、预防疾病&#xff0c;以更饱满的精神状态拥抱生活。 合理饮食是养生的基石。《黄帝内经》中提到 “五谷为养&#xff0c;五果为助&#xff0c;五畜为益&…...

深入理解Java阻塞队列:原理、使用场景及代码实战

&#x1f680; 文章提示 你将在这篇文章中收获&#xff1a; 阻塞队列的核心特性&#xff1a;队列空/满时的阻塞机制 四种操作方式对比&#xff1a;抛异常、返回特殊值、永久阻塞、超时阻塞 SynchronousQueue的独特设计&#xff1a;同步队列的生产者-消费者强耦合 代码实战&a…...

vue3--手写手机屏组件

<!--* 手机预览* Author: Hanyang* Date: 2022-12-09 09:13:00* LastEditors: Hanyang* LastEditTime: 2023-01-12 15:37:00 --> <template><divclass"public-preview-mobile"ref"previewMobileRef":class"showMobile ? animation-sh…...

【Elasticsearch】入门篇

Elasticsearch 入门 前言 官方地址&#xff1a;Elastic — 搜索 AI 公司 | Elastic ES 下载地址&#xff1a;Past Releases of Elastic Stack Software | Elastic 文档&#xff1a;什么是 Elasticsearch&#xff1f;|Elasticsearch 指南 简介 Elasticsearch 是一个分布式、…...

Unity 使用 ADB 实时查看手机运行性能

Unity 使用 ADB 实时查看手机运行性能 前言操作步骤ADB工具下载ADB工具配置手机进入开发者模式并开启USB调试使用ADB连接手机Unity打包设置使用Profiler实时查看性能情况优化建议 常见问题 前言 通过 ADB&#xff08;Android Debug Bridge&#xff09;连接安卓设备&#xff0c…...

蓝桥杯 1. 四平方和

四平方和 原题目链接 题目描述 四平方和定理&#xff08;又称拉格朗日定理&#xff09;指出&#xff1a; 每个正整数都可以表示为 至多 4 个正整数的平方和。 如果将 0 包括进去&#xff0c;则每个正整数都可以恰好表示为 4 个非负整数的平方和。 例如&#xff1a; 5 0 …...

Nginx 配置参数全解版:Nginx 反向代理与负载均衡;Nginx 配置规范与 Header 透传实践指南;Nginx 配置参数详解

Nginx 配置参数全解版&#xff1a;Nginx 反向代理与负载均衡&#xff1b;Nginx 配置规范与 Header 透传实践指南&#xff1b;Nginx 配置参数详解 Nginx 反向代理与负载均衡配置&#xff0c;Header 透传到后端应用&#xff08;参数全解版&#xff09;一、Nginx 反向代理与负载均…...

数据分析之技术干货业务价值​​ powerquery 分组排序后取TOP

在电商中&#xff0c;我们要对货品进行分析&#xff0c;由于所有的销售数据都在一起&#xff0c;货品信息也在一起&#xff0c;两个表建立了关系之后&#xff0c;要看每个品类的TOP款有哪些&#xff0c;每个品类的TOP款是什么要怎么做呢&#xff1f; 下面是我做数据的思路&…...

windows中kafka4.0集群搭建

参考文献 Apache Kafka windows启动kafka4.0&#xff08;不再需要zookeeper&#xff09;_kafka压缩包-CSDN博客 Kafka 4.0 KRaft集群部署_kafka4.0集群部署-CSDN博客 正文 注意jdk需要17版本以上的 修改D:\software\kafka_2.13-4.0.0\node1\config\server.properties配置文…...

数据分析案例:医疗健康数据分析

目录 数据分析案例:医疗健康数据分析1. 项目背景2. 数据加载与预处理2.1 加载数据2.2 数据清洗3. 探索性数据分析(EDA)3.1 再入院率概览3.2 按年龄分组的再入院率3.3 住院时长与再入院4. 特征工程与可视化5. 模型构建与评估5.1 数据划分5.2 训练逻辑回归5.3 模型评估6. 业务…...

数据分析之 商品价格分层之添加价格带

在分析货品数据的时候&#xff0c;我们会对商品的价格进行分层汇总&#xff0c;也叫价格带&#xff0c;​​ 一、价格带的定义​​ ​​价格带&#xff08;Price Band&#xff09;​​&#xff1a;将商品按价格区间划分&#xff08;如0-50元、50-100元、100-200元等&#xff…...

跨浏览器音频录制:实现兼容的音频捕获与WAV格式生成

在现代Web开发中&#xff0c;音频录制功能越来越受到开发者的关注。无论是在线会议、语音识别还是简单的语音留言&#xff0c;音频录制都是一个重要的功能。然而&#xff0c;实现一个跨浏览器的音频录制功能并非易事&#xff0c;因为不同浏览器对音频录制API的支持存在差异。本…...

JavaScript 的“世界模型”:深入理解对象 (Objects)

引言&#xff1a;超越简单值&#xff0c;构建复杂实体 到目前为止&#xff0c;我们学习的变量大多存储的是单一的值&#xff0c;比如一个数字 (let age 30;​)、一个字符串 (let name "Alice";​) 或一个布尔值 (let isActive true;​)。这对于简单场景足够了&am…...

【国产化之路】VPX-3U :基于D2000 /FT2000的硬件架构到操作系统兼容

在国产化和高性能计算、嵌入式系统领域日益受到重视的今天&#xff0c;VPX3U架构以其标准化和模块化的特性广受关注。本文将从硬件架构、系统软件、接口拓展及典型应用等方面&#xff0c;深入剖析整体设计思路与工程实现&#xff0c;供友友们参考和讨论。 一、总体架构与设计目…...

深入探索RAG(检索增强生成)模型的优化技巧

&#x1f4cc; 友情提示&#xff1a; 本文内容由银河易创AI&#xff08;https://ai.eaigx.com&#xff09;创作平台的gpt-4o-mini模型生成&#xff0c;旨在提供技术参考与灵感启发。文中观点或代码示例需结合实际情况验证&#xff0c;建议读者通过官方文档或实践进一步确认其准…...

架构-软件架构设计

一、软件架构基础概念 1. 软件架构的定义 通俗理解&#xff1a;软件架构是软件系统的“骨架”&#xff0c;定义了系统的结构、行为和属性&#xff0c;就像盖房子的设计图纸&#xff0c;规划了房间布局、承重结构和功能分区。核心作用&#xff1a; 沟通桥梁&#xff1a;让技术…...

免费的 HTML 网页托管服务

字根云平台最近上线了一项新的服务&#xff1a; HTML 网页托管。 HTML 网页免费托管 免费HTML静态页面文件托管-字根秀秀​ www.cuobiezi.net/showshow/ 网页托管主要支持的功能&#xff1a; 1. 托管静态的 HTML 网页&#xff08;不包含图片&#xff0c;图片请使用专业图床&a…...

Ubuntu服务器上如何监控Oracle数据库

在 Ubuntu 服务器上监控 Oracle 数据库&#xff0c;虽然不像在 Windows 或某些企业 Linux&#xff08;如 RHEL&#xff09;那样有现成的 GUI 工具&#xff0c;但你完全可以通过命令行工具、脚本、开源监控平台来实现全面监控&#xff0c;包含&#xff1a; 数据库性能指标&#…...

什么是CMMI认证?CMMI评估内容?CMMI认证能带来哪些好处?

CMMI认证详解&#xff1a;概念、评估内容与核心价值 一、什么是CMMI认证&#xff1f; CMMI&#xff08;Capability Maturity Model Integration&#xff0c;能力成熟度模型集成&#xff09;是由美国卡内基梅隆大学软件工程研究所&#xff08;SEI&#xff09;开发的一套全球公…...

PySide6 GUI 学习笔记——常用类及控件使用方法(常用类矩阵QRectF)

文章目录 类描述构造方法主要方法1. 基础属性2. 边界操作3. 几何运算4. 坐标调整5. 转换方法6. 状态判断 类特点总结1. 浮点精度&#xff1a;2. 坐标系统&#xff1a;3. 有效性判断&#xff1a;4. 几何运算&#xff1a;5. 类型转换&#xff1a;6. 特殊处理&#xff1a; 典型应用…...

arm64适配系列文章-第五章-arm64环境上redis的部署

ARM64适配系列文章 第一章 arm64环境上kubesphere和k8s的部署 第二章 arm64环境上nfs-subdir-external-provisioner的部署 第三章 arm64环境上mariadb的部署 第四章 arm64环境上nacos的部署 第五章 arm64环境上redis的部署 第六章 arm64环境上rabbitmq-management的部署 第七章…...

Ubuntu / WSL 安装pipx

一、安装pipx 在 Ubuntu / WSL 上可以用两种方式安装&#xff0c;推荐第二种&#xff08;官方脚本&#xff09;或第三种&#xff08;pip 安装最新版本并自动配置 PATH&#xff09;。 1. apt 安装&#xff08;最快&#xff0c;但版本往往偏旧&#xff09; sudo apt update su…...

10天学会嵌入式技术之51单片机-day-6

第十五章 点阵LED 15.1 点阵 LED 概述 15.1.1 实物图 15.1.1 原理图 15.2 点阵 LED 静态显示 15.2.1 需求描述 使用点阵 LED 显示一排由左上到右下的斜线&#xff0c;具体效果如下图所示。 15.2.2 硬件设计 15.2.2.1 硬件原理图 根据内部原理可知&#xff0c;点阵 LED 的…...

【数据可视化-28】2017-2025 年每月产品零售价数据可视化分析

&#x1f9d1; 博主简介&#xff1a;曾任某智慧城市类企业算法总监&#xff0c;目前在美国市场的物流公司从事高级算法工程师一职&#xff0c;深耕人工智能领域&#xff0c;精通python数据挖掘、可视化、机器学习等&#xff0c;发表过AI相关的专利并多次在AI类比赛中获奖。CSDN…...

【JavaScript】`Object` 对象静态方法详解

在 JavaScript 中,Object 对象提供了许多静态方法,用于操作和处理对象。以下是一些常用的 Object.xxx 方法及其用途和示例: 1. 属性相关方法 Object.keys(obj) 返回对象自身的所有可枚举属性的键组成的数组。 const obj = {a: 1, b: 2 }; console.log(Object.keys(obj)); //…...

音视频之H.265/HEVC量化

H.265/HEVC系列文章&#xff1a; 1、音视频之H.265/HEVC编码框架及编码视频格式 2、音视频之H.265码流分析及解析 3、音视频之H.265/HEVC预测编码 4、音视频之H.265/HEVC变换编码 5、音视频之H.265/HEVC量化 量化 (Quantization) 是指将信号的连续取值(或大量可能的离散取…...

Tomcat:从零理解Java Web应用的“心脏”

目录 一、Tomcat是什么&#xff1f;为什么需要它&#xff1f; 二、Tomcat的核心架构&#xff08;餐厅运营图&#xff09; 1. 两大核心组件 2. 请求处理全流程&#xff08;从点餐到上菜&#xff09; 三、手把手搭建第一个网站&#xff08;厨房开张实录&#xff09; 环境准备…...

第七届能源系统与电气电力国际学术会议(ICESEP 2025)

重要信息 时间&#xff1a;2025年6月20-22日 地点&#xff1a;中国-武汉 官网&#xff1a;www.icesep.net 主题 能源系统 节能技术、能源存储技术、可再生能源、热能与动力工程 、能源工程、可再生能源技术和系统、风力发…...

【基础】Node.js 介绍、安装及npm 和 npx功能了解

前言 后面安装n8n要用到&#xff0c;做一点技术储备。主要是它的两个工具&#xff1a;npm 和 npx。 Node.js介绍 Node.js 是一个免费的、开源的、跨平台的 JavaScript 运行时环境&#xff0c;允许开发人员在浏览器之外编写命令行工具和服务器端脚本&#xff0c;是一个基于 C…...

【硬核干货】SonarQube安全功能

原文链接&#xff1a;【硬核干货】SonarQube安全功能 关于晓数神州 晓数神州坚持以“客户为中心”的宗旨&#xff0c;为客户提供专业的解决方案和技术服务&#xff0c;构建多引擎数字化体系。 核心业务1&#xff1a;聚焦DevOps全栈产品&#xff0c;打造需求管理、项目管理、开…...

微信小程序 tabbar底部导航栏

官方文档&#xff1a;https://developers.weixin.qq.com/miniprogram/dev/reference/configuration/app.html#tabBar 一、常规菜单格式 在app.json 文件中配置&#xff0c;其他关键点详见官方文档&#xff0c;后续更新不规则图标的写法...

如何将极狐GitLab 议题导出为 CSV?

极狐GitLab 是 GitLab 在中国的发行版&#xff0c;关于中文参考文档和资料有&#xff1a; 极狐GitLab 中文文档极狐GitLab 中文论坛极狐GitLab 官网 导出议题到 CSV (BASIC ALL) 您可以将问题从极狐GitLab 导出为 CSV 文件&#xff0c;这些文件将作为附件发送到您的默认通知…...

nodejs之Express-介绍、路由

五、Express 1、express 介绍 express 是一个基于 Node.js 平台的极简、灵活的 WEB 应用开发框架,官方网址: https://www.expressjs.com.cn/ 简单来说,express 是一个封装好的工具包,封装了很多功能,便于我们开发 WEB 应用(HTTP 服务) (1)基本使用 第一步:初始化项目并…...

极狐GitLab 如何从 CSV 导入议题?

极狐GitLab 是 GitLab 在中国的发行版&#xff0c;关于中文参考文档和资料有&#xff1a; 极狐GitLab 中文文档极狐GitLab 中文论坛极狐GitLab 官网 从 CSV 导入议题 (BASIC ALL) 您可以通过上传包含以下列的 CSV 文件将议题导入项目&#xff1a; 名称是否必需&#xff1f;…...

JW01三合一传感器详解(STM32)

目录 一、介绍 二、传感器原理 1.原理图 2.引脚描述 三、程序设计 main文件 usart3.h文件 usart3.c文件 四、实验效果 五、资料获取 项目分享 一、介绍 JW01三合一检测模块是一种用于检测空气中二氧化碳浓度的传感器模块。它可以广泛应用于室内空气质量检测、智能家…...

23种设计模式-行为型模式之策略模式(Java版本)

Java 策略模式&#xff08;Strategy Pattern&#xff09;详解 &#x1f9e0; 什么是策略模式&#xff1f; 策略模式是一种行为型设计模式&#xff0c;它定义了一系列算法&#xff0c;把它们一个个封装起来&#xff0c;并且使它们可以互相替换。策略模式让算法独立于使用它的客…...

ActiveMQ 快速上手:安装配置与基础通信实践(一)

一、引言 在当今分布式系统和微服务架构盛行的时代&#xff0c;消息通信作为实现系统间解耦、异步处理和可靠传输的关键技术&#xff0c;显得尤为重要。ActiveMQ 作为一款广泛应用的开源消息中间件&#xff0c;凭借其对 JMS 规范的全面支持、丰富的特性以及出色的性能&#xf…...

究竟什么是自动化测试?

自动化测试是一种软件测试方法&#xff0c;旨在通过使用自动化工具和脚本来执行测试任务&#xff0c;以减少人工操作&#xff0c;提高测试效率和准确性。 以下是对自动化测试的详细介绍&#xff1a; 一、定义与特点 定义&#xff1a;自动化测试是指利用自动化工具和脚本来执…...

【LLM+Code】Github Copilot Agent/VsCode Agent 模式PromptTools详细解读

一、前言 github copilot agent mode现在和vscode是强绑定的关系&#xff0c; 其实是一个东西&#xff1a; https://github.blog/news-insights/product-news/github-copilot-the-agent-awakens/https://code.visualstudio.com/docs/copilot/chat/chat-agent-mode 二、Syste…...

IDEA将本地的JAR文件手动安装到 Maven的本地仓库

例如这是要导入的依赖&#xff1a; mvn install:install-file -DfileD:\aliyun-java-sdk-ding.jar -DgroupIdcom.aliyun -DartifactIdaliyun-java-sdk-ding -Dversion1.0.0 -Dpackagingjar-DfileD:\aliyun-java-sdk-ding.jar 含义&#xff1a;指定要安装到本地 Maven 仓库的 …...

redis集群的三种部署方式

一、主从同步 redis的主从同步工作原理简单概括为: 1、从服务器(Slave Server)向(主服务器,Master)发送sync(同步)命令 2、master启动后台存盘进程,并收集所有修改数据命令 3、master完成存盘后,传送整个数据文件到slave 4、slave接受数据文件,加载到内存中完成首次…...

【GIT】github中的仓库如何删除?

你可以按照以下步骤删除 GitHub 上的仓库&#xff08;repository&#xff09;&#xff1a; &#x1f6a8; 注意事项&#xff1a; ❗️删除仓库是不可恢复的操作&#xff0c;所有代码、issue、pull request、release 等内容都会被永久删除。 &#x1f9ed; 删除 GitHub 仓库步骤…...

CIFAR10图像分类学习笔记(三)---数据加载load_cifar10

新创建一个load_cifar10源文件 需要导入的包 import glob from torchvision import transforms from torch.utils.data import DataLoader ,Dataset import os #读取工具 from PIL import Image import numpy as np 01同样定义10个类别的标签名数组 label_name ["airpl…...

基于Matlab的车牌识别系统

1.程序简介 本模型基于MATLAB,通过编程创建GUI界面&#xff0c;基于Matlab的数字图像处理&#xff0c;对静止的车牌图像进行分割并识别&#xff0c;通过编写matlab程序对图像进行灰度处理、二值化、腐蚀膨胀和边缘化处理等&#xff0c;并定位车牌的文字&#xff0c;实现字符的…...