Spring Boot 与 RabbitMQ 的深度集成实践(四)
实战案例
业务场景描述
在电商系统中,用户下单是一个核心业务操作。当用户成功下单后,系统需要执行一系列后续任务,如发送邮件通知用户订单已成功提交,更新库存信息以确保商品库存的准确性,以及记录订单相关的日志信息等。
在传统的实现方式中,这些任务通常是在下单操作完成后同步执行的。例如,在订单服务中,当用户下单后,代码会依次调用邮件发送服务的接口来发送邮件通知,调用库存服务的接口来更新库存,以及调用日志服务的接口来记录订单日志。这种同步执行的方式存在一些明显的问题:
- 系统响应速度慢:由于每个后续任务都需要等待前一个任务完成才能执行,而发送邮件、更新库存等操作可能涉及网络请求、数据库读写等耗时操作,这会导致下单操作的响应时间延长。如果用户在下单后需要等待较长时间才能得到响应,可能会降低用户体验,甚至导致用户流失。
- 系统耦合度高:订单服务与邮件服务、库存服务、日志服务之间存在紧密的耦合关系。订单服务需要知道各个服务的接口细节和调用方式,并且这些服务的任何变动(如接口参数调整、服务地址变更等)都可能影响到订单服务的正常运行,增加了系统的维护难度和风险。
- 系统扩展性差:当业务需求发生变化,需要添加新的后续任务(如发送短信通知)或修改现有任务的执行逻辑时,订单服务的代码需要进行较大的改动,这不利于系统的扩展和升级。
为了解决这些问题,可以引入 RabbitMQ 消息队列来实现异步处理和解耦。具体实现方式如下:
- 订单消息的产生:当用户在电商系统上下单后,订单服务会创建一个包含订单详细信息(如订单编号、用户信息、商品列表、下单时间等)的订单消息,并将该消息发送到 RabbitMQ 的订单队列中。订单服务在发送消息后,无需等待后续任务的执行结果,即可立即返回给用户下单成功的响应,大大提高了系统的响应速度。
- 邮件通知任务:邮件服务作为消费者,监听 RabbitMQ 中的订单队列。当有新的订单消息到达时,邮件服务从队列中获取订单消息,根据消息中的用户信息和订单内容,生成邮件内容并调用邮件发送接口,将邮件发送给用户。这样,邮件发送任务与订单服务解耦,订单服务无需关心邮件发送的具体实现和细节。
- 库存更新任务:库存服务同样监听订单队列。获取订单消息后,根据消息中的商品信息和数量,更新库存数据库中的商品库存数量。如果库存不足,还可以触发相应的补货流程或通知管理员。通过这种方式,库存更新任务与订单服务分离,提高了系统的可维护性和扩展性。
- 日志记录任务:日志服务监听订单队列,获取订单消息后,将订单相关信息记录到日志数据库中,以便后续进行数据分析、故障排查等操作。
通过引入 RabbitMQ 实现异步处理和解耦后,电商系统的下单流程变得更加高效、灵活和可靠。订单服务专注于处理订单的核心逻辑,而后续任务由各个独立的服务异步处理,降低了系统组件之间的耦合度,提高了系统的整体性能和可维护性。同时,RabbitMQ 的消息持久化、消息确认等机制保证了订单消息的可靠传递和处理,避免了因系统故障导致的任务丢失或数据不一致问题。
代码实现与演示
假设电商系统使用 Spring Boot 构建,以下是生产者发送订单消息和消费者处理订单消息的具体代码示例:
生产者发送订单消息代码:
import org.springframework.amqp.rabbit.core.RabbitTemplate;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
@Service
public class OrderProducer {
@Autowired
private RabbitTemplate rabbitTemplate;
public void sendOrder(Order order) {
// 将订单对象转换为JSON格式字符串发送
rabbitTemplate.convertAndSend("order.exchange", "order.routing.key", order);
System.out.println("Sent order: " + order);
}
}
在上述代码中,OrderProducer类通过注入RabbitTemplate来发送订单消息。sendOrder方法接收一个Order对象作为参数,该对象包含订单的详细信息。通过rabbitTemplate.convertAndSend方法,将订单对象转换为 JSON 格式的字符串,并发送到名为order.exchange的交换机,使用order.routing.key作为路由键,这样消息就会被路由到对应的队列中。
消费者处理订单消息代码:
import org.springframework.amqp.rabbit.annotation.RabbitListener;
import org.springframework.stereotype.Component;
@Component
public class OrderConsumer {
@RabbitListener(queues = "order.queue")
public void handleOrder(Order order) {
try {
// 处理订单逻辑,如更新库存、发送邮件通知等
System.out.println("Received order: " + order);
// 模拟更新库存操作
updateInventory(order);
// 模拟发送邮件通知操作
sendEmailNotification(order);
System.out.println("Order processed successfully.");
} catch (Exception e) {
System.out.println("Error processing order: " + e.getMessage());
}
}
private void updateInventory(Order order) {
// 实际业务中调用库存服务更新库存
System.out.println("Updating inventory for order: " + order);
}
private void sendEmailNotification(Order order) {
// 实际业务中调用邮件服务发送邮件
System.out.println("Sending email notification for order: " + order);
}
}
在OrderConsumer类中,使用@RabbitListener注解监听名为order.queue的队列。当队列中有新的订单消息时,handleOrder方法会被触发。在该方法中,首先打印接收到的订单信息,然后调用updateInventory方法模拟更新库存操作,调用sendEmailNotification方法模拟发送邮件通知操作。如果在处理订单过程中发生异常,会捕获异常并打印错误信息。
运行结果演示:
- 启动 Spring Boot 应用程序,确保 RabbitMQ 服务器也处于运行状态。
- 调用订单服务的下单接口,创建一个订单并发送订单消息。例如,可以通过 Postman 发送 HTTP 请求到订单服务的下单接口,请求体中包含订单的详细信息。
- 在订单生产者的控制台输出中,可以看到类似以下的信息,表示订单消息已成功发送:
Sent order: Order{orderId='123456', userId='user1', productList=[Product{productId='p1', quantity=2}, Product{productId='p2', quantity=1}], orderTime=2024-10-28T15:30:00}
- 在订单消费者的控制台输出中,可以看到类似以下的信息,表示订单消息已被成功接收并处理:
Received order: Order{orderId='123456', userId='user1', productList=[Product{productId='p1', quantity=2}, Product{productId='p2', quantity=1}], orderTime=2024-10-28T15:30:00}
Updating inventory for order: Order{orderId='123456', userId='user1', productList=[Product{productId='p1', quantity=2}, Product{productId='p2', quantity=1}], orderTime=2024-10-28T15:30:00}
Sending email notification for order: Order{orderId='123456', userId='user1', productList=[Product{productId='p1', quantity=2}, Product{productId='p2', quantity=1}], orderTime=2024-10-28T15:30:00}
Order processed successfully.
通过以上代码实现和运行结果演示,可以清晰地看到 Spring Boot 与 RabbitMQ 集成后,订单消息的发送和处理过程。这种集成方式有效地实现了系统的异步处理和解耦,提高了系统的性能和可维护性。
常见问题与解决方案
连接问题
在集成 Spring Boot 与 RabbitMQ 的过程中,连接问题是较为常见的。其中,连接超时是一个典型的问题,当 Spring Boot 应用尝试连接 RabbitMQ 服务器时,如果在规定的时间内未能成功建立连接,就会抛出连接超时异常。这可能是由于网络不稳定,例如网络延迟过高、网络中断等,导致应用无法及时与服务器进行通信;也可能是 RabbitMQ 服务器负载过高,无法及时响应新的连接请求。
解决连接超时问题,可以在配置文件中适当增加连接超时时间。在 Spring Boot 中,可以通过配置CachingConnectionFactory的connectionTimeout属性来实现。例如,在application.yml中添加如下配置:
spring:
rabbitmq:
host: localhost
port: 5672
username: guest
password: guest
virtual - host: /
connection - factory:
connection - timeout: 10000 # 设置连接超时时间为10秒
这样可以给连接过程更多的时间来完成,从而减少因短暂网络波动或服务器繁忙导致的连接超时。
拒绝访问也是常见的连接问题之一,通常是由于权限不足或配置错误导致的。例如,使用了错误的用户名或密码,或者用户没有足够的权限访问指定的虚拟主机。要解决这个问题,首先需要确认在application.yml中配置的用户名和密码是否正确,并且该用户在 RabbitMQ 中具有访问指定虚拟主机的权限。可以通过 RabbitMQ 的管理界面(如http://localhost:15672),在 “Users” 页面检查用户的权限设置,确保用户对相应的虚拟主机有 “Configure”、“Write” 和 “Read” 权限。如果权限不足,可以在管理界面中为用户添加所需的权限。
消息丢失和重复消费
消息丢失是一个严重的问题,它可能发生在消息生产、传输和消费的各个环节。在生产环节,消息可能因为网络问题未能成功发送到 RabbitMQ 服务器。为了避免这种情况,可以启用生产者确认机制(Confirm Callback)。在application.yml中配置publisher - confirm - type: correlated,开启发布确认模式,然后在RabbitTemplate中设置ConfirmCallback回调。当消息成功发送到交换机时,ack参数为true,否则为false,并可以通过cause参数获取失败原因,以便进行相应的处理,如记录日志或重新发送消息 。
在消息从交换机路由到队列的过程中,如果路由键不正确或者没有匹配的队列,消息也可能丢失。可以启用 Return 回调机制来捕获这种情况。在application.yml中设置publisher - returns: true,并在RabbitTemplate中设置ReturnsCallback回调。当消息无法路由到队列时,会触发该回调,通过returnedMessage参数可以获取到未路由的消息、交换机、路由键等信息,从而进行相应的处理,如将消息存储到备份队列或记录错误日志。
消费者在处理消息时,如果发生异常并且没有进行正确的处理,也可能导致消息丢失。可以将消费者的确认模式设置为手动确认,在application.yml中配置spring.rabbitmq.listener.simple.acknowledge - mode: manual。这样,消费者在成功处理消息后,需要手动调用channel.basicAck方法来确认消息;如果处理过程中出现异常,则调用channel.basicNack方法拒绝消息,并可以根据业务需求决定是否将消息重新放入队列 。
重复消费也是一个常见的问题,通常是由于网络波动、消费者故障或确认机制异常等原因导致的。当消费者处理完消息但在向 RabbitMQ 发送确认消息时出现网络问题,RabbitMQ 没有收到确认,就会认为消息未被消费,从而再次发送该消息,导致重复消费。
为了解决重复消费问题,可以引入幂等性处理。幂等性是指对同一操作进行多次执行所产生的影响与一次执行的影响相同。在业务逻辑中,可以通过数据库的唯一约束、状态机等方式来实现幂等性。例如,在处理订单消息时,可以在数据库中为订单表的某个唯一标识字段(如订单编号)添加唯一约束。当消费者接收到订单消息并处理时,先根据订单编号查询数据库中是否已经存在该订单记录。如果存在,则说明该订单已经被处理过,直接返回,不再重复处理;如果不存在,则进行正常的订单处理流程,并将订单信息插入数据库 。还可以利用分布式缓存(如 Redis)来实现幂等性。在消费者处理消息前,先根据消息的唯一标识(如消息 ID)在 Redis 中查询是否已经处理过该消息。如果已经存在该标识,则说明消息已经被处理,直接返回;如果不存在,则将消息标识存入 Redis,并进行消息处理。
性能优化
从队列设计方面来看,合理设置队列的属性可以提高性能。例如,根据业务需求选择合适的队列类型,如果业务场景中需要广播消息,扇形队列可能更合适;如果需要根据路由规则进行消息分发,直连队列或主题队列会更适用。设置队列的持久化属性时,要权衡性能和可靠性。持久化队列会将数据写入磁盘,虽然保证了数据的可靠性,但会影响写入性能。对于一些对可靠性要求不高但对性能要求较高的场景,可以选择非持久化队列 。
在消息批量处理方面,Spring AMQP 提供了批量发送和接收消息的支持。在生产者端,可以使用RabbitTemplate的send方法的批量版本,将多个消息一次性发送到 RabbitMQ 服务器,减少网络开销。在消费者端,可以配置SimpleRabbitListenerContainerFactory的batchSize属性,使消费者每次从队列中获取多个消息进行批量处理。例如:
import org.springframework.amqp.rabbit.config.SimpleRabbitListenerContainerFactory;
import org.springframework.amqp.rabbit.connection.ConnectionFactory;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration
public class RabbitMQConfig {
@Bean
public SimpleRabbitListenerContainerFactory rabbitListenerContainerFactory(ConnectionFactory connectionFactory) {
SimpleRabbitListenerContainerFactory factory = new SimpleRabbitListenerContainerFactory();
factory.setConnectionFactory(connectionFactory);
factory.setBatchSize(10); // 设置每次批量获取10条消息
return factory;
}
}
在资源配置方面,合理分配服务器资源对于 RabbitMQ 的性能至关重要。要确保服务器有足够的内存和 CPU 资源来处理大量的消息。如果服务器内存不足,可能会导致消息堆积在队列中,影响系统性能;如果 CPU 使用率过高,会导致消息处理速度变慢。可以通过监控工具(如 Prometheus 和 Grafana)实时监控 RabbitMQ 服务器的资源使用情况,根据监控数据调整服务器的配置,如增加内存、升级 CPU 等。还可以对 RabbitMQ 进行集群部署,通过负载均衡将消息分发到多个节点上进行处理,提高系统的整体性能和可用性 。
总结与展望
通过本文的深入探讨,我们全面地了解了 Spring Boot 与 RabbitMQ 的深度集成实践。从技术背景来看,Spring Boot 凭借其简化开发流程、自动配置等特性,为 Java 开发者提供了高效便捷的开发体验;RabbitMQ 基于 AMQP 协议,以其灵活的路由机制、可靠的消息传递能力,成为分布式系统中消息通信的重要支撑 。
在集成过程中,我们详细阐述了从环境搭建、项目创建到配置 RabbitMQ 连接信息,再到创建消息队列、交换机以及编写消息生产者和消费者的各个步骤。通过这些步骤,实现了 Spring Boot 应用与 RabbitMQ 的无缝对接,使得系统能够利用消息队列进行高效的异步通信和解耦。同时,我们还深入研究了消息持久化、消息确认机制、死信队列和延迟队列等高级特性的实现,这些特性进一步提升了消息传递的可靠性和灵活性,满足了各种复杂业务场景的需求。
在电商系统下单流程的实战案例中,我们清晰地看到了 Spring Boot 与 RabbitMQ 集成在实际业务中的应用效果。通过引入消息队列,实现了订单处理与后续任务的异步处理和解耦,显著提高了系统的响应速度和可维护性,充分体现了这种集成方式在解决实际业务问题中的强大优势。
展望未来,随着分布式系统和微服务架构的不断发展,Spring Boot 与 RabbitMQ 的集成将在更多领域和场景中得到广泛应用。在金融领域,可用于实现交易系统中的异步通知、风险监控等功能;在物联网领域,能够处理大量设备产生的实时数据,实现设备与后端系统之间的可靠通信。未来的研究可以朝着进一步优化性能、增强安全性以及探索更多创新应用场景的方向发展,例如结合云计算技术,实现消息队列的弹性扩展和高可用性,为企业的数字化转型提供更强大的技术支持。
参考资料
- Spring Boot 官方文档
- RabbitMQ 官方文档
- 《Spring Boot 实战》,Craig Walls 著,涵盖 Spring Boot 的基础与进阶知识,对理解 Spring Boot 原理和应用有很大帮助。
- Spring Boot 与 RabbitMQ 集成教程,提供了详细的 Spring Boot 与 RabbitMQ 集成示例和说明。
相关文章:
Spring Boot 与 RabbitMQ 的深度集成实践(四)
实战案例 业务场景描述 在电商系统中,用户下单是一个核心业务操作。当用户成功下单后,系统需要执行一系列后续任务,如发送邮件通知用户订单已成功提交,更新库存信息以确保商品库存的准确性,以及记录订单相关的日志信…...
ES6详解
一、变量声明 let 与 const 块级作用域:替代 var 的函数作用域 const 声明常量(不可重新赋值,但对象属性可修改) if (true) {let x 10const PI 3.14 } console.log(x) // 报错 二、箭头函数 简写语法与 this 绑定 // 传统函数…...
C语言—字符函数和字符串函数
1.字符分类函数 字符控制函数:int iscntrl ( int c ); 控制字符通常不是可打印字符,该函数是用来判断参数是否为控制字符,需要的头文件为<ctype.h>标准ASCII码中,不可打印字符主要包括以下两类: 控制字符&…...
【LeetCode】大厂面试算法真题回忆(93)--优雅数组
题目描述 如果一个数组中出现次数最多的元素出现大于等于k次,被称为k-优雅数组,k也可以被称为优雅阈值。 例如,数组[1, 2, 3, 1, 2, 3, 1],它是一个3-优雅数组,因为元素1出现次数大于等于3次。数组[1, 2, 3, 1, 2]就不是一个3-优雅数组,因为其中出现次数最多的元素是1和…...
【MySQL成神之路】MySQL常用语法总结
目录 MySQL 语法总结 数据库操作 表操作 数据操作 查询语句 索引操作 约束 事务控制 视图操作 存储过程和函数 触发器 用户和权限管理 数据库操作 创建数据库: CREATE DATABASE database_name; 选择数据库: USE database_name; 删除数…...
机器学习第十六讲:K-means → 自动把超市顾客分成不同消费群体
机器学习第十六讲:K-means → 自动把超市顾客分成不同消费群体 资料取自《零基础学机器学习》。 查看总目录:学习大纲 关于DeepSeek本地部署指南可以看下我之前写的文章:DeepSeek R1本地与线上满血版部署:超详细手把手指南 K-me…...
多商户1.8.1版本前端问题优化集合指南
1、逛逛社区上传一张图时,进入详情页面显示不出来 修改路径:pages ---> discover ---> components ---> discoverDetails.vue 解读:这里是因为图片高度没有定义,图片没显示出来。修改如下: <!--逛逛类型为…...
基于正点原子阿波罗F429开发板的LWIP应用(1)——网络ping通
说在开头 正点原子F429开发板主芯片采用的是STM32F429IGT6,网络PHY芯片采用的是LAN8720A(V1)和YT8512C(V2),采用的是RMII连接,PHY_ADDR为0;在代码中将会对不同的芯片做出适配。 CubeMX版本:6.6.1; F4芯片组…...
第 1 章:数字 I/O 与串口通信(GPIO UART)
本章目标: 掌握 GPIO 的硬件原理、寄存器配置与典型驱动框架 深入理解 UART/USART 的帧格式、波特率配置、中断与 DMA 驱动 通过实战案例,将 GPIO 与 UART 结合,实现 AT 命令式外设控制 章节结构 GPIO 概述与硬件原理 GPIO 驱动实现:寄存器、中断与去抖 UART/USART 原理与帧…...
MCU 温度采样理论(-ADC Temperature sensor)
温度传感器可以使用ADC来测量芯片温度。 为了准确测量运行时的芯片温度,请使用在生产过程中运行的参考测量值,此参考值与其他校准数据一起存放在SFlash中。 一、温度测量流程 1、ADC校准:关于偏移和增益调整的实例,见9.3。 2、检查CREFH和VREL:参见8.2。 3、设置参考…...
stm32week16
stm32学习 十一.中断 4.使用中断 EXTI的配置步骤: 使能GPIO时钟设置GPIO输入模式使能AFIO/SYSCFG时钟设置EXTI和IO对应关系设置EXTI屏蔽,上/下沿设置NVIC设计中断服务函数 HAL库的使用: 使能GPIO时钟:__HAL_RCC_GPIOx_CLK_EN…...
隨筆 20250519 基于MAUI Blazor整合SQLite数据库与Star打印机的详细步骤
以下是基于MAUI Blazor整合SQLite数据库与Star打印机的详细步骤,包含必要的NuGet包引入及核心代码实现: 零、目錄結構 一、整合SQLite数据库 1. 安装NuGet包 # SQLite核心库 Install-Package sqlite-net-pcl # SQLite平台适配库&am…...
电子电路原理第十六章(负反馈)
1927年8月,年轻的工程师哈罗德布莱克(Harold Black)从纽约斯塔顿岛坐渡轮去上班。为了打发时间,他粗略写下了关于一个新想法的几个方程式。后来又经过反复修改, 布莱克提交了这个创意的专利申请。起初这个全新的创意被认为像“永动机”一样愚蠢可笑,专利申请也遭到拒绝。但…...
推客小程序系统开发:全栈式技术解决方案与行业赋能实践
在数字化营销深度渗透各行业的当下,传统推广模式已难以满足企业精细化运营与高效获客的需求。专业的推客小程序系统凭借其强大的裂变传播能力与灵活的推广机制,成为企业构建私域流量池、提升推广效能的核心工具。我们基于多年技术沉淀与行业洞察&…...
【prometheus+Grafana篇】基于Prometheus+Grafana实现Oracle数据库的监控与可视化
💫《博主主页》: 🔎 CSDN主页 🔎 IF Club社区主页 🔥《擅长领域》:擅长阿里云AnalyticDB for MySQL(分布式数据仓库)、Oracle、MySQL、Linux、prometheus监控;并对SQLserver、NoSQL(MongoDB)有了…...
【Android构建系统】Soong构建系统,通过.bp + .go定制编译
背景介绍 本篇是一篇实操内容,是对【Android构建系统】如何在Camera Hal的Android.bp中选择性引用某个模块的优化与改进。本篇内容主要想通过一个具体例子介绍Soong构建系统较复杂的定制化方法和步骤,以便在今后的工作学习中更好的使用Soong构建系统。 …...
Qt开发:QUdpSocket的详解
文章目录 一、QUdpSocket 简介二、常用函数的介绍和使用三、接收端完整示例四、发送端完整示例 一、QUdpSocket 简介 在 Qt 中,UDP(User Datagram Protocol,用户数据报协议)是通过 QUdpSocket 类实现的。UDP 是一种无连接的、轻量…...
【android bluetooth 协议分析 01】【HCI 层介绍 9】【ReadLocalSupportedCommands命令介绍】
1. HCI_Read_Local_Supported_Commands 命令介绍 1. 命令介绍(Description) HCI_Read_Local_Supported_Commands 是 HCI 层中非常重要的查询命令。它允许 Host(如 Android 系统中的 Bluetooth stack)获取 Controller(…...
Model 速通系列(一)nanoGPT
这个是新开的一个系列用来手把手复现一些模型工程,之所以开这个系列是因为有人留言说看到一个工程不知道从哪里读起,出于对自身能力的提升与兴趣,故新开了这个系列。由于主要动机是顺一遍代码并提供注释。 该系列第一篇博客是 nanoGPT &…...
星际争霸小程序:用Java实现策略模式的星际大战
在游戏开发的世界里,策略模式是一种非常实用的设计模式,它允许我们在运行时动态地选择算法或行为。今天,我将带你走进一场星际争霸的奇幻之旅,用Java实现一个简单的星际争霸小程序,通过策略模式来模拟不同种族单位的战…...
网络Tips20-007
网络威胁会导致非授权访问、信息泄露、数据被破坏等网络安全事件发生, 其常见的网络威胁包括窃听、拒绝服务、病毒、木马、( 数据完整性破坏 )等, 常见的网络安全防范措施包括访问控制、审计、身份认证、数字签名、( 数据加密 )、 包过滤和检测等。 AE…...
2.微服务-配置
引入springcloud的pom配置 <parent><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-parent</artifactId><version>2.7.12</version><relativePath/></parent> <dependencyManagemen…...
python实现pdf转图片(针对每一页)
from pdf2image import convert_from_path import ospdf_file rC:\Users\\Desktop\拆分\产权证.pdf poppler_path rC:\poppler-24.08.0\Library\bin # 这里改成你自己的路径output_dir rC:\Users\\Desktop\拆分\output_images os.makedirs(output_dir, exist_okTrue)image…...
Python编程从入门到实践 PDF 高清版
各位程序员朋友们,还在为找不到合适的Python学习资料而烦恼吗?还在为晦涩难懂的编程书籍而头疼吗?今天,就给大家带来一份重磅福利——237完整版PDF, 我用网盘分享了「Python编程:从入门到实践__超清版.pdf…...
CVE-2015-3934 Fiyo CMS SQL注入
CVE-2015-3934 Fiyo CMS SQL注入 页面抓登录数据包 构造延时注入语句在user处’%2B(select(0)from(select(sleep(5)))v)%2B’ 存在延时注入,使用脚本即可...
【Pandas】pandas DataFrame mode
Pandas2.2 DataFrame Computations descriptive stats 方法描述DataFrame.abs()用于返回 DataFrame 中每个元素的绝对值DataFrame.all([axis, bool_only, skipna])用于判断 DataFrame 中是否所有元素在指定轴上都为 TrueDataFrame.any(*[, axis, bool_only, skipna])用于判断…...
(思维题、贪心)洛谷 P11232 CSPS2024 超速检测 题解
这一题在 2024 将我击败,但我怎么现在才补题解 …… 题意 原题 思路 对于每一辆车,我们可以算出,其在距离左端点哪段位置会超速 [ l , r ] [l,r] [l,r],那么这辆车会被 l l l 右侧最近的测速仪到 r r r 左侧最近的测速仪检…...
C#:多线程
一.线程常用概念 线程(Thread):操作系统执行程序的最小单位 进程(Process):程序在内存中的运行实例 并发(Concurrency):多个任务交替执行(单核CPU࿰…...
虚拟币制度钱包开发:功能设计与成本全解析
虚拟币制度钱包开发:功能设计与成本全解析 ——从基础架构到合规风控的完整解决方案 一、开发成本:分层定价与关键影响因素 根据2024-2025年行业数据显示,虚拟币钱包App开发成本跨度较大,主要受功能复杂度、技术架构与合规要求三…...
TransmittableThreadLocal实现上下文传递-笔记
1.TransmittableThreadLocal简介 com.alibaba.ttl.TransmittableThreadLocal(简称 TTL)是阿里巴巴开源的一个工具类,旨在解决 ThreadLocal 在线程池中无法传递上下文变量 的问题。它是对 InheritableThreadLocal 的增强,尤其适用…...
应对WEEE 2025:猎板PCB的区块链追溯与高温基材创新
在全球电子产业加速向循环经济转型的背景下,欧盟《绿色新政》与《WEEE指令》对PCB行业提出更高要求。作为行业先行者,猎板PCB(Hunter PCB)以生物降解基材为核心,结合全球合规体系与产业链协同创新,构建从材…...
大陆资产在香港发行RWA的合规路径与核心限制
大陆资产在香港发行RWA的合规路径与核心限制 ——从“双重合规原则”到资产准入边界的全景解读 一、法律框架:双重合规原则的刚性约束 根据香港金管局Ensemble沙盒项目要求,大陆资产在香港发行RWA需遵循“双重合规原则”,即底层资产需同时符…...
爬虫攻防战:从入门到放弃的完整对抗史与实战解决方案
爬虫攻防战:从入门到放弃的完整对抗史与实战解决方案 这张有趣的图片生动描绘了爬虫开发者与反爬工程师之间的"军备竞赛"。作为技术博主,我将基于这张图的各个阶段,深入分析爬虫技术的演进与对应的反制措施,提供一套完整的反爬解决方案,包括技术原理、实施方法…...
Fabric初体验(踩坑笔记)
搭建fabric部署合约学习笔记 环境准备CURl安装docker 参照官网文档实现(2025.05.19)根据前言交代的文章去尝试(失败版)安装fabric-samples安装指定2.2.0版本Fabric二进制文件和配置文件直接手动下载(不建议)…...
区块链blog2_中心化与效率
🌿中心化出现原因 信息/服务分散在各处会浪费时间且不方便使用,由此,把信息/服务集中在一起,便于管理,避免了不必要的效率损失。 即集中资源,使得对信息处理的全过程效率升高。中心化不是网络中产生的&…...
2024年ASOC SCI2区TOP,多机制群优化算法+多风场输电线路巡检中多无人机任务分配与路径规划,深度解析+性能实测
目录 1.摘要2.考虑风场影响的多无人机任务分配3.基于双向蚁群和离散蜜獾算法求解多无人机任务分配问题(BACOHBA)4.考虑风场的多无人机路径规划5.结果展示6.参考文献7.代码获取8.读者交流 1.摘要 随着电力系统规模的不断扩大,复杂环境下的电力线路及设施的巡检与维护…...
智慧赋能光伏运维——无人机巡检+地面监控双链路覆盖,打造光伏电站管理新标杆
一、引言:光伏电站运维的挑战与机遇 在全球能源转型浪潮下,光伏电站作为清洁能源的重要载体,其高效运维管理成为行业核心命题。然而,传统光伏电站运维存在覆盖范围广、设备分散、人工巡检效率低、故障响应慢等痛点。为破解这一难…...
c/c++的opencv开闭操作
OpenCV 中的形态学开运算与闭运算 (C) 在计算机视觉和图像处理领域,形态学操作是用于分析和处理图像形状的一系列非线性操作。OpenCV 作为一个强大的开源计算机视觉库,提供了丰富的形态学转换函数。其中,“开运算”(Opening&…...
Linux利用多线程和线程同步实现一个简单的聊天服务器
1. 概述 本文实现一个基于TCP/IP的简单多人聊天室程序。它包含一个服务器端和一个客户端:服务器能够接收多个客户端的连接,并将任何一个客户端发来的消息广播给所有其他连接的客户端;客户端则可以连接到服务器,发送消息并接收来自…...
无人机遥控器光纤通信模块技术要点!
一、技术要点 1. 长距离低损耗传输 采用单模光纤(如G.654.E光纤),利用光纤的低衰减特性(0.17 dB/km以下),支持10公里以上的远距离通信,突破了传统无线信号因衰减导致的覆盖限制。例如&…...
深入解析OkHttp与Retrofit:Android网络请求的黄金组合
前言 在移动应用开发中,网络请求是连接客户端与服务器的关键桥梁。对于Android开发者而言,OkHttp和Retrofit这对组合已经成为处理网络请求的事实标准。本文将全面剖析这两个框架的设计理念、核心功能、协同关系以及最佳实践,帮助开发者构建高…...
Python操作PDF书签详解 - 添加、修改、提取和删除
目录 简介 使用工具 Python 向 PDF 添加书签 添加书签 添加嵌套书签 Python 修改 PDF 书签 Python 展开或折叠 PDF 书签 Python 提取 PDF 书签 Python 删除 PDF 书签 简介 PDF 书签是 PDF 文件中的导航工具,通常包含一个标题和一个跳转位置(如…...
Spring Boot与Kafka集成实践:从入门到实战
Spring Boot与Kafka集成实践 引言 在现代分布式系统中,消息队列是不可或缺的组件之一。Apache Kafka作为一种高吞吐量的分布式消息系统,广泛应用于日志收集、流处理、事件驱动架构等场景。Spring Boot作为Java生态中最流行的微服务框架,提供…...
luckysheet的使用——17.将表格作为pdf下载到本地
luckysheet源码里面自带有打印按钮,但是功能是无法使用的,所以我把该功能重写了一遍 1.在menuButton.js文件中找到源码打印按钮的触发事件: $("#luckysheet-icon-print").click(function () {}2.使用自己写的挂载方法 window.pr…...
矿井支架LCYVB-6钢丝编织护套连接器介绍
LCYVB-6钢丝编织护套连接器是一种专为矿井支架设计的连接装置,主要用于增强支架的稳定性和安全性。该连接器采用高强度钢丝编织护套,具有优异的抗拉强度和耐磨性,适用于恶劣的矿井环境。 主要特点 高强度钢丝编织护套:采用优质钢…...
git仓库中.git 文件很大,怎么清理掉一部分
查询 .git 文件大小,在 git-bash 里执行(后面有些命令不能执行,也请在 git-bash 里执行) windows11 安装好后右键没有 git bash 命令-CSDN博客 du -sh .git // 592m .git 操作前最好先备份一份,避免推送到远程时出错…...
Qt框架核心组件完全指南:从按钮交互到定时器实现
文章目录 前言一、QAbstractButton 按钮类概述1.1 常用属性1.2 常用信号1.3QButtonGroup 按钮组 二、QComboBox 组合框三、若干与数字相关的组件四、QString 字符串类五、Qt容器类5.1 顺序容器 QList5.2 关联容器 QMap 六、QVariant七、跨平台数据类型7.1 基础数据类型7.2 特殊…...
Axure设计数字乡村可视化大屏:从布局到交互的实战经验分享
乡村治理正从传统模式向“数据驱动”转型。数字乡村可视化大屏作为数据展示的核心载体,不仅能直观呈现乡村发展全貌,还能为决策提供科学依据。本文以Axure为工具,结合实际案例,分享如何从零设计一个功能完备、交互流畅的数字乡村大…...
60天python训练计划----day30
DAY 30 模块和库的导入 知识点回顾: 导入官方库的三种手段导入自定义库/模块的方式导入库/模块的核心逻辑:找到根目录(python解释器的目录和终端的目录不一致) 一.导入官方库 我们复盘下学习python的逻辑,所谓学习pyth…...
HJ3 明明的随机数【牛客网】
文章目录 零、原题链接一、题目描述二、测试用例三、解题思路3.1 快排去重3.2 散列 四、参考代码4.1 快排去重4.2 散列 零、原题链接 HJ3 明明的随机数 一、题目描述 二、测试用例 三、解题思路 3.1 快排去重 基本思路: 先将序列进行快速排序,然后…...