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

Spring Boot 集成 RabbitMQ

在现代分布式系统中,消息队列扮演着至关重要的角色。它能够实现系统间的异步通信、解耦组件以及提高系统的可扩展性和可靠性。RabbitMQ 作为一款广泛使用的开源消息中间件,具有强大的功能和灵活的配置。而 Spring Boot 则是一种流行的 Java 开发框架,能够快速构建应用程序。本文将详细介绍如何在 Spring Boot 项目中集成 RabbitMQ,包括安装和配置 RabbitMQ、在 Spring Boot 中使用 RabbitMQ 的步骤以及实际应用案例。

一、引言

随着软件系统的规模和复杂性不断增加,传统的同步通信方式已经无法满足需求。消息队列作为一种异步通信机制,可以有效地解耦系统之间的依赖关系,提高系统的可扩展性和可靠性。RabbitMQ 以其高可靠性、高吞吐量和灵活的路由机制,成为了许多企业级应用的首选消息中间件。Spring Boot 则提供了一种快速、便捷的方式来构建应用程序,使得开发者可以更加专注于业务逻辑的实现。将 Spring Boot 与 RabbitMQ 集成,可以充分发挥两者的优势,构建出高效、可靠的消息通信系统。

二、RabbitMQ 基础概念

(一)什么是 RabbitMQ

RabbitMQ 是一个开源的消息代理和队列服务器,实现了高级消息队列协议(AMQP)。它提供了可靠的消息传递、灵活的路由机制和高可用性,适用于各种分布式系统和微服务架构。

(二)RabbitMQ 的核心概念

  1. 消息(Message)
    • 消息是 RabbitMQ 中传递的数据单元。它可以包含任何类型的数据,如文本、二进制数据、JSON 等。消息由生产者发送到 RabbitMQ 服务器,并由消费者从服务器中接收和处理。
  2. 生产者(Producer)
    • 生产者是发送消息的应用程序。它将消息发送到 RabbitMQ 服务器中的特定队列或交换器。
  3. 消费者(Consumer)
    • 消费者是接收和处理消息的应用程序。它从 RabbitMQ 服务器中的队列中获取消息,并对其进行处理。
  4. 队列(Queue)
    • 队列是存储消息的容器。生产者将消息发送到队列中,消费者从队列中获取消息进行处理。队列可以有不同的名称和属性,如持久性、排他性等。
  5. 交换器(Exchange)
    • 交换器是用于将消息路由到不同队列的组件。生产者将消息发送到交换器,交换器根据一定的规则将消息路由到一个或多个队列中。
  6. 绑定(Binding)
    • 绑定是将队列与交换器关联起来的操作。通过绑定,交换器可以将消息路由到特定的队列中。

(三)RabbitMQ 的工作原理

  1. 生产者发送消息
    • 生产者将消息发送到 RabbitMQ 服务器中的特定交换器。交换器根据消息的路由键和绑定规则,将消息路由到一个或多个队列中。
  2. 消费者接收消息
    • 消费者从队列中获取消息进行处理。消费者可以通过订阅队列来接收消息,也可以通过主动拉取的方式从队列中获取消息。
  3. 消息确认
    • 消费者在处理完消息后,需要向 RabbitMQ 服务器发送消息确认。如果消费者在处理消息过程中出现异常,或者在一定时间内没有发送消息确认,RabbitMQ 服务器会将消息重新发送给其他消费者进行处理。

三、安装和配置 RabbitMQ

(一)安装 RabbitMQ

  1. 在 Linux 系统上安装 RabbitMQ
    • 可以通过以下步骤在 Linux 系统上安装 RabbitMQ:
      • 添加 RabbitMQ 软件源:

sudo apt-get install -y software-properties-common
sudo add-apt-repository -y "deb http://www.rabbitmq.com/debian/ testing main"

  • 导入 RabbitMQ 公钥:

wget -O- https://www.rabbitmq.com/rabbitmq-release-signing-key.asc | sudo apt-key add -

  • 更新软件包列表:

sudo apt-get update

  • 安装 RabbitMQ 服务器:

sudo apt-get install -y rabbitmq-server

  1. 在 Windows 系统上安装 RabbitMQ
    • 可以从 RabbitMQ 官方网站下载 Windows 版本的安装程序,然后按照安装向导进行安装。

(二)配置 RabbitMQ

  1. 启动 RabbitMQ 服务器
    • 在 Linux 系统上,可以使用以下命令启动 RabbitMQ 服务器:

sudo service rabbitmq-server start

  • 在 Windows 系统上,可以在安装目录中找到 RabbitMQ 服务,然后启动该服务。

  1. 创建用户和虚拟主机
    • 可以使用 RabbitMQ 的管理插件来创建用户和虚拟主机。在浏览器中输入http://localhost:15672,进入 RabbitMQ 的管理界面。然后,使用默认的用户名和密码(guest/guest)登录。
    • 在管理界面中,可以创建新的用户和虚拟主机,并为用户分配相应的权限。
  2. 配置交换器和队列
    • 可以使用 RabbitMQ 的管理界面或者编程方式来配置交换器和队列。交换器和队列的配置取决于具体的应用需求,可以设置不同的名称、类型、属性等。

四、Spring Boot 集成 RabbitMQ 的步骤

(一)添加依赖

在 Spring Boot 项目的pom.xml文件中添加以下依赖:

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

这个依赖将引入 Spring Boot 对 RabbitMQ 的支持。

(二)配置 RabbitMQ

application.propertiesapplication.yml文件中添加 RabbitMQ 的配置信息:

spring.rabbitmq.host=localhost
spring.rabbitmq.port=5672
spring.rabbitmq.username=your_username
spring.rabbitmq.password=your_password
spring.rabbitmq.virtual-host=your_virtual_host

这些配置信息指定了 RabbitMQ 服务器的地址、端口、用户名、密码和虚拟主机。可以根据实际情况进行修改。

(三)创建生产者

  1. 创建一个生产者配置类,用于配置生产者的属性:

import org.springframework.amqp.core.Binding;
import org.springframework.amqp.core.BindingBuilder;
import org.springframework.amqp.core.Queue;
import org.springframework.amqp.core.TopicExchange;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;@Configuration
public class RabbitMQProducerConfig {public static final String QUEUE_NAME = "my_queue";public static final String EXCHANGE_NAME = "my_exchange";public static final String ROUTING_KEY = "my_routing_key";@Beanpublic Queue queue() {return new Queue(QUEUE_NAME, false, false, false);}@Beanpublic TopicExchange exchange() {return new TopicExchange(EXCHANGE_NAME);}@Beanpublic Binding binding() {return BindingBuilder.bind(queue()).to(exchange()).with(ROUTING_KEY);}
}

在这个配置类中,我们创建了一个队列、一个交换器和一个绑定。队列的名称为my_queue,交换器的名称为my_exchange,路由键为my_routing_key
2. 创建一个生产者服务类,用于发送消息:

import org.springframework.amqp.rabbit.core.RabbitTemplate;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;@Service
public class RabbitMQProducerService {@Autowiredprivate RabbitTemplate rabbitTemplate;public void sendMessage(String message) {rabbitTemplate.convertAndSend(RabbitMQProducerConfig.EXCHANGE_NAME, RabbitMQProducerConfig.ROUTING_KEY, message);}
}

这个服务类使用RabbitTemplate来发送消息。可以在其他地方注入这个服务类,并调用sendMessage方法来发送消息。

(四)创建消费者

  1. 创建一个消费者配置类,用于配置消费者的属性:

import org.springframework.amqp.core.Binding;
import org.springframework.amqp.core.BindingBuilder;
import org.springframework.amqp.core.Queue;
import org.springframework.amqp.core.TopicExchange;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;@Configuration
public class RabbitMQConsumerConfig {public static final String QUEUE_NAME = "my_queue";public static final String EXCHANGE_NAME = "my_exchange";public static final String ROUTING_KEY = "my_routing_key";@Beanpublic Queue queue() {return new Queue(QUEUE_NAME, false, false, false);}@Beanpublic TopicExchange exchange() {return new TopicExchange(EXCHANGE_NAME);}@Beanpublic Binding binding() {return BindingBuilder.bind(queue()).to(exchange()).with(ROUTING_KEY);}
}

这个配置类与生产者配置类类似,创建了相同的队列、交换器和绑定。
2. 创建一个消费者服务类,用于处理接收到的消息:

import org.springframework.amqp.rabbit.annotation.RabbitListener;
import org.springframework.stereotype.Service;@Service
public class RabbitMQConsumerService {@RabbitListener(queues = RabbitMQConsumerConfig.QUEUE_NAME)public void consumeMessage(String message) {System.out.println("Received message: " + message);}
}

这个服务类使用@RabbitListener注解来定义一个消费者方法,该方法将在接收到消息时被调用。可以根据实际需求对消息进行处理。

五、Spring Boot 集成 RabbitMQ 的配置项

(一)生产者配置项

  1. spring.rabbitmq.template.exchange:生产者发送消息时使用的交换器名称。
  2. spring.rabbitmq.template.routing-key:生产者发送消息时使用的路由键。
  3. spring.rabbitmq.template.mandatory:当消息无法路由到任何队列时,是否返回给生产者。默认值为false

(二)消费者配置项

  1. spring.rabbitmq.listener.simple.acknowledge-mode:消费者确认消息的方式。可选值有manual(手动确认)和auto(自动确认)。默认值为auto
  2. spring.rabbitmq.listener.simple.prefetch:消费者每次从队列中获取的消息数量。默认值为1
  3. spring.rabbitmq.listener.simple.concurrency:消费者的并发数量。默认值为1

六、Spring Boot 集成 RabbitMQ 的实际应用案例

(一)异步任务处理

  1. 场景描述
    • 在一个 Web 应用程序中,用户提交了一个耗时的任务,如文件上传、数据处理等。为了提高用户体验,可以将这个任务放入消息队列中,由后台的异步任务处理服务从消息队列中获取任务并进行处理。
  2. 实现步骤
    • 当用户提交任务时,Web 应用程序将任务信息发送到 RabbitMQ 服务器中的特定队列中。
    • 创建一个异步任务处理服务,使用 Spring Boot 集成 RabbitMQ 的消费者功能,从队列中获取任务信息并进行处理。处理完成后,可以将结果发送回 Web 应用程序或者存储到数据库中。

(二)系统间通信

  1. 场景描述
    • 在一个分布式系统中,不同的系统之间需要进行通信。可以使用 RabbitMQ 作为消息中间件,实现系统间的异步通信和解耦。
  2. 实现步骤
    • 各个系统将需要发送的消息发送到 RabbitMQ 服务器中的特定队列中。
    • 接收消息的系统使用 Spring Boot 集成 RabbitMQ 的消费者功能,从队列中获取消息并进行处理。可以根据消息的类型和内容进行相应的业务处理。

(三)事件驱动架构

  1. 场景描述
    • 在一个事件驱动架构中,系统中的各个组件通过发布和订阅事件来进行通信。可以使用 RabbitMQ 作为事件总线,实现事件的发布和订阅。
  2. 实现步骤
    • 当一个组件发生了一个事件时,它将事件信息发送到 RabbitMQ 服务器中的特定交换器中。交换器根据事件的类型和路由键,将事件路由到一个或多个队列中。
    • 订阅了相应事件的组件使用 Spring Boot 集成 RabbitMQ 的消费者功能,从队列中获取事件信息并进行处理。可以根据事件的内容进行相应的业务处理。

七、性能优化和故障排除

(一)性能优化

  1. 调整 RabbitMQ 服务器配置
    • 根据实际情况调整 RabbitMQ 服务器的配置参数,如内存分配、磁盘空间、网络参数等,以提高 RabbitMQ 的性能。
  2. 优化生产者和消费者代码
    • 在生产者和消费者代码中,避免不必要的序列化和反序列化操作,减少网络传输开销。
    • 合理设置生产者的重试次数和超时时间,以提高消息发送的成功率和性能。
    • 对于消费者,可以根据实际情况调整拉取消息的频率和批量处理的大小,以提高消费效率。
  3. 使用多个队列和交换器
    • 根据业务需求合理划分消息的类型和路由规则,使用多个队列和交换器来提高消息的处理效率和可扩展性。

(二)故障排除

  1. 消息丢失或重复
    • 检查生产者和消费者的配置参数,确保消息的发送和消费过程正确。
    • 检查 RabbitMQ 服务器的配置参数,确保消息的持久化和确认机制正常工作。
    • 如果出现消息丢失或重复的情况,可以通过调整生产者和消费者的配置参数,或者使用 RabbitMQ 的事务功能来保证消息的一致性。
  2. 消费延迟
    • 检查消费者的拉取频率和批量处理大小,是否设置合理。
    • 检查 RabbitMQ 服务器的负载情况,是否存在性能瓶颈。
    • 如果消费延迟较高,可以考虑增加消费者的数量,或者调整 RabbitMQ 服务器的配置参数,以提高消费效率。
  3. 连接问题
    • 检查 RabbitMQ 服务器的地址和端口是否正确配置。
    • 检查网络连接是否正常,是否存在防火墙等限制。
    • 如果出现连接问题,可以通过检查网络配置、调整防火墙规则等方式来解决。

八、总结

本文详细介绍了如何在 Spring Boot 项目中集成 RabbitMQ,包括安装和配置 RabbitMQ、在 Spring Boot 中使用 RabbitMQ 的步骤以及实际应用案例。通过集成 RabbitMQ,我们可以构建出高效、可靠的消息通信系统,实现系统间的异步通信和解耦。在实际应用中,我们还可以根据需要进行性能优化和故障排除,以确保系统的稳定运行。希望本文对大家在 Spring Boot 集成 RabbitMQ 方面有所帮助。

相关文章:

Spring Boot 集成 RabbitMQ

在现代分布式系统中&#xff0c;消息队列扮演着至关重要的角色。它能够实现系统间的异步通信、解耦组件以及提高系统的可扩展性和可靠性。RabbitMQ 作为一款广泛使用的开源消息中间件&#xff0c;具有强大的功能和灵活的配置。而 Spring Boot 则是一种流行的 Java 开发框架&…...

存在sql注入的公网站点

此数据为博主在新手阶段练习sql注入时发现的站点&#xff0c;漏洞可能修复&#xff0c;备注可能错误 url: https://www.uni-1.com.hk/about_en.php?id2 注入点类型&#xff1a;数值 sql报错回显&#xff1a;无 sql语句执行&#xff1a;[order by] [union] 字段数&#xff1a;1…...

linux之网络子系统- 内核发送数据包流程以及相关实际问题

一、相关实际问题 查看内核发送数据消耗的CPU时应该看sy还是si在服务器上查看/proc/softirqs&#xff0c;为什么NET_RX要比NET_TX大得多发送网络数据的时候都涉及那些内存拷贝操作零拷贝到底是怎么回事为什么Kafka的网络性能很突出 二、网络包发送过程总览 调用系统调用send发…...

UDP 实现的 Echo Server 和 Echo Client 回显程序

欢迎浏览高耳机的博客 希望我们彼此都有更好的收获 感谢三连支持! 在网络编程中&#xff0c;Echo Server 和 Echo Client 回显是一种经典的示例&#xff0c;用于演示基本的网络通信。Echo Server 接收来自客户端的数据&#xff0c;并将相同的数据发送回客户端。这种模式在测试…...

AUTOSAR CP MCAL微控制器抽象层介绍

AUTOSAR&#xff08;Automotive Open System Architecture&#xff09;即汽车开放系统架构&#xff0c;它将汽车电子控制单元&#xff08;ECU&#xff09;的软件底层做了一个标准的封装&#xff0c;使得开发者能够共用一套底层软件&#xff0c;并通过修改参数来匹配不同的硬件和…...

SpringBoot应用部署到Docker中MySQL8时间戳相差8小时问题及处理方式

文章目录 SpringBoot应用部署到Docker中MySQL8时间戳相差8小时问题及处理方式1. 检查MySQL服务器的时间区设置2. 在Spring Boot应用程序中设置时间区3. Docker容器中通过Dockerfile设置时区4. 在运行Docker容器时通过命令行传递环境变量5. 启动SpringBoot应用时设置JVM参数来指…...

飞桨首创 FlashMask :加速大模型灵活注意力掩码计算,长序列训练的利器

在 Transformer 类大模型训练任务中&#xff0c;注意力掩码&#xff08;Attention Mask&#xff09;一方面带来了大量的冗余计算&#xff0c;另一方面因其 O ( N 2 ) O(N^2) O(N2)巨大的存储占用导致难以实现长序列场景的高效训练&#xff08;其中 N N N为序列长度&#xff09;…...

【含文档+源码】基于SpringBoot+Vue的新型吃住玩一体化旅游管理系统的设计与实现

开题报告 本文旨在探讨新型吃住玩一体化旅游管理系统的设计与实现。该系统融合了用户注册与登录、旅游景点管理、旅游攻略发帖、特色旅游路线推荐、附近美食推荐以及酒店客房推荐与预定等多项功能&#xff0c;旨在为游客提供全方位、一体化的旅游服务体验。在系统设计中&#…...

【网络安全】揭示 Web 缓存污染与欺骗漏洞

未经许可,不得转载。 文章目录 前言污染与欺骗Web 缓存污染 DoS1、HTTP 头部超大 (HHO)2、HTTP 元字符 (HMC)3、HTTP 方法覆盖攻击 (HMO)4、未键入端口5、重定向 DoS6、未键入头部7、Host 头部大小写规范化8、路径规范化9、无效头部 CP-DoS10、HTTP 请求拆分Web 缓存污染与有害…...

PHP如何防止防止源代码的暴露

在PHP开发中&#xff0c;防止源代码暴露是确保应用程序安全性的重要一环。源代码暴露可能会让攻击者发现敏感信息&#xff0c;如数据库凭据、业务逻辑漏洞等&#xff0c;从而进行恶意攻击。以下是一些防止PHP源代码暴露的方法&#xff1a; 禁用PHP短标签&#xff1a; 在php.in…...

C++智能指针的实现

本篇文章详细探讨下如何使用裸指针实现智能指针。 补充内容 由于本篇文章主要是探讨怎么实现三种智能指针,但是在编码过程中,博主可能会使用些有些同学不了解的特性,为了保证大家思绪不被打断,博主先把这些小特性介绍出来,大家选择性参考。 1、什么是RAII? RAII(Reso…...

硅谷(12)菜单管理

菜单管理模块 11.1 模块初始界面 11.1.1 API&&type API: import request from /utils/request import type { PermisstionResponseData, MenuParams } from ./type //枚举地址 enum API {//获取全部菜单与按钮的标识数据ALLPERMISSTION_URL /admin/acl/permission…...

定子调压调速系统

定子调压调速系统是一种用于控制三相交流绕线电机的调速系统&#xff0c;它通过改变电动机定子电压和转子电阻来实现对电机转速的控制。以下是关于定子调压调速系统的详细介绍&#xff1a; 工作原理 定子电压调控&#xff1a;在1&#xff5e;3档时&#xff0c;系统通过控制定子…...

从APP小游戏到Web漏洞的发现

一、前因&#xff1a; 在对一次公司的一个麻将游戏APP进行渗透测试的时候发现&#xff0c;抓到HTTP请求的接口&#xff0c;但是反编译APK后发现没有在本身发现任何一个关于接口或者域名相关的关键字&#xff0c;对此感到了好奇。 于是直接解压后everything搜索了一下&#xff…...

设计模式07-结构型模式(装饰模式/外观模式/代理模式/Java)

4.4 装饰模式 4.4.1 装饰模式的定义 1.动机&#xff1a;在不改变一个对象本身功能的基础上给对象增加额外的新行为 2.定义&#xff1a;动态地给一个对象增加一些额外的职责&#xff0c;就增加对象功能来说&#xff0c;装饰模式比生成子类实现更为灵活 4.4.2 装饰模式的结构…...

C# 广播技术——发现局域网设备技术——

一广播技术应用 客户端发现与管理&#xff1a;软件可以通过广播消息来发现网络中的客户端&#xff0c;从而方便对客户端进行集中管理和监控。服务通知&#xff1a;向所有客户端广播重要的通知、更新或警告信息&#xff0c;确保客户端及时了解相关情况。资源共享与分配&#xf…...

【QA】windows和linux陷入系统调用后有什么区别?

最近被某面试官的这个问题拷打&#xff0c;当场脸烧起来… 首先讲讲系统调用&#xff1a; 系统调用是操作系统为调用者提供服务的接口&#xff0c;以便程序员聚焦于业务问题。分为文件操作&#xff0c;内存分配&#xff0c;进程管理等等。用户使用系统调用后会触发软中断&…...

Github 2024-11-01 开源项目月报 Top19

根据Github Trendings的统计,本月(2024-11-01统计)共有19个项目上榜。根据开发语言中项目的数量,汇总情况如下: 开发语言项目数量Python项目9TypeScript项目3JavaScript项目3Svelte项目1Jupyter Notebook项目1Ruby项目1HTML项目1Rust项目1Java项目1C++项目1Go项目1Python中的…...

Python实现深度学习模型预测控制(tensorflow)DL-MPC(Deep Learning Model Predictive Control

链接&#xff1a;深度学习模型预测控制 &#xff08;如果认为有用&#xff0c;动动小手为我点亮github小星星哦&#xff09;&#xff0c;持续更新中…… 链接&#xff1a;WangXiaoMingo/TensorDL-MPC&#xff1a;DL-MPC&#xff08;深度学习模型预测控制&#xff09;是基于 P…...

Anki插件Export deck to html的改造

在Anki中进行复习时&#xff0c;每次只能打开一条笔记。如果积累了很多笔记&#xff0c;有时候会有将它们集中输出成一个pdf进行阅读的想法。Anki插件Export deck to html&#xff08;安装ID&#xff1a;1897277426&#xff09;就有这个功能。但是&#xff0c;这个插件目前存在…...

csdn 记载文章十分缓慢

现象&#xff1a;其它网站能够正常访问&#xff0c;但只要点击某个 csdn 页面就会等很久 &#xff08;本文只针对 csdn 网页&#xff0c;其他网页出现类似情况可以同样处理&#xff09; 产生这种现象的原因&#xff1a; 这种情况很有可能是在访问 CSDN 主页时&#xff0c;需要向…...

python通过pyperclip库操作剪贴板

pyperclip介绍 pyperclip是一个python库用于操作剪贴板&#xff0c;可以非常方便地将文本复制到剪贴板或从剪贴板获取文本。 通过pip进行安装&#xff1a;pip install pyperclip pyperclip的github地址 pyperclip使用 复制到剪贴板 import pypercliptext "Hello, Wo…...

LSTM——长短期记忆神经网络

目录 1.LSTM 工作原理 2.LSTM的代码实现 3.代码详解 LSTM&#xff08;Long Short-Term Memory&#xff09;是一种特殊的循环神经网络&#xff08;RNN&#xff09;&#xff0c;用于解决长序列中的长期依赖问题。它通过引入门机制&#xff0c;控制信息的流入、保留和输出&…...

10进阶篇:运用第一性原理解答“是什么”类型题目

在667分析题题型中,关于“如何做”和“好处是什么”的题目,许多同学都能较好地运用前述的667作答地图开展答题,但是唯独在“是什么”类型题目(也可以叫做认识型题目),不知从何下手。这种题目通常要求我们理解、分析,并展望未来的发展方向,而结构化、逻辑清晰的答案往往…...

【elkb】索引生命周期管理

索引生命周期管理 Index lifecycle management(索引生命周期管理)是elasticsearch提供的一种用于自动管理索引的生命周期的功能。允许使用者定义索引的各个阶段&#xff0c;从创建至删除。并允许使用者在每个阶段定义索引需要执行的特定动作。这些动作包含索引创建&#xff0c…...

江协科技STM32学习- P25 UART串口协议

&#x1f680;write in front&#x1f680; &#x1f50e;大家好&#xff0c;我是黄桃罐头&#xff0c;希望你看完之后&#xff0c;能对你有所帮助&#xff0c;不足请指正&#xff01;共同学习交流 &#x1f381;欢迎各位→点赞&#x1f44d; 收藏⭐️ 留言&#x1f4dd;​…...

15分钟学 Go 第 22 天:包的使用

第22天&#xff1a;包的使用 欢迎来到Go语言的第22天&#xff01;今天&#xff0c;我们将深入探讨如何创建和使用包。通过学习包的使用&#xff0c;你将能够更好组织你的代码&#xff0c;提高复用性和可维护性。 1. 包的概念 在Go语言中&#xff0c;包是代码的基本组织单位。…...

【Leecode】Leecode刷题之路第35天之搜索插入位置

题目出处 35-搜索插入位置-题目出处 题目描述 个人解法 思路&#xff1a; 1.依次遍历数组&#xff0c;看目标值是否在数组中 2.如果不在&#xff0c;将目标值插入数组&#xff08;涉及到数组移动、扩容&#xff09;&#xff0c;返回下标代码示例&#xff1a;&#xff08;Java…...

速盾:海外cdn高防

随着互联网的快速发展&#xff0c;网站的安全性和稳定性变得越来越重要。尤其是对于大型企业和电商平台来说&#xff0c;保护用户数据和维护网站稳定运行是至关重要的。为了应对日益增长的网络攻击和恶意访问&#xff0c;海外CDN高防服务成为了一种非常受欢迎的解决方案。 首先…...

图书管理系统(JDBC)

AdminUser是管理员类 NormalUser是用户类 AddOperation是增加图书类 BorrowOperation是借书类 DelOperation是删除图书类 ExitOperation是退出类 FindOperation是查找图书类 IOPeration是接口 ReturnOperation是还书类 ShowOperation是显示所有图书类 注意&#xff1a…...