SpringBoot整合RabbitMq企业级使用封装

SpringBoot整合RabbitMq企业级使用封装

  • 1、RabbitMq基础说明
  • 2、SpringBoot整合RabbitMq,以及RabbitMq的封装和高级用法
    • 2.1、pom.xml
    • 2.2、application.yml
    • 2.3、Mq配置类MessageQueueConfiguration
      • 2.3.1、代码
      • 2.3.2、设置replyTimeout
      • 2.3.3、publisher-confirm-type和mandatory
    • 2.4、自定义发送消息帮助类MessageQueueHelper
    • 2.4、消费者
    • 2.5、两个pojo
    • 2.6、Controller
  • 3、SpringBoot整合RabbitMq,用SimpleMessageListenerContainer更复杂业务的封装
  • 4、死信队列和延迟队列

1、RabbitMq基础说明

本文主要是企业级的SpringBoot整合RabbitMq,妥妥的符合任务使用MQ的业务配置。

下面实际讲的就是高级部分,rabbitmq的初级部分没说,也就分为以下几个点,你理解即可:
1、消息应答,就是下面的ACK机制,分为自动应答和手动应答。
2、发布确认,就是下面ConfirmCallback,ReturnCallback这两个回调函数的执行场景。
3、交换机:分为3中Fanout,Direct ,Topic 。随便百度理解下概念,一般最常用的就是Topic ,因为Topic 扩展性强,下面案例中也就是使用的Topic 。

如果下面案例你真正理解了,那么你的MQ可以适用于任何复杂业务的封装场景了。

2、SpringBoot整合RabbitMq,以及RabbitMq的封装和高级用法

说明:下面这个例子,是把邮件和短信放到MQ里,然后消费端去消费。
列子中列举了MQ的所有的高级特性,具体看下代码注释,很详细。

2.1、pom.xml

    <dependencies><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId><version>2.7.5</version></dependency><dependency><groupId>org.mybatis.spring.boot</groupId><artifactId>mybatis-spring-boot-starter</artifactId></dependency><dependency><groupId>mysql</groupId><artifactId>mysql-connector-java</artifactId></dependency><dependency><groupId>org.projectlombok</groupId><artifactId>lombok</artifactId></dependency><!--整合rabbitMq--><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-amqp</artifactId><version>2.7.5</version></dependency></dependencies>

2.2、application.yml

server:port: 8080spring:datasource:driver-class-name: com.mysql.cj.jdbc.Driverusername: rootpassword: rooturl: jdbc:mysql://127.0.0.1:3306/rightcloud?useUnicode=true&autoReconnect=true&failOverReadOnly=false&characterEncoding=UTF-8&allowMultiQueries=true&useSSL=false&serverTimezone=Asia/Shanghai&zeroDateTimeBehavior=CONVERT_TO_NULLrabbitmq:host: 127.0.0.1port: 5672username: guestpassword: guestrequested-heartbeat: 60spublisher-confirm-type: correlatedpublisher-returns: true

2.3、Mq配置类MessageQueueConfiguration

2.3.1、代码

import lombok.extern.slf4j.Slf4j;
import org.springframework.amqp.core.AcknowledgeMode;
import org.springframework.amqp.rabbit.config.SimpleRabbitListenerContainerFactory;
import org.springframework.amqp.rabbit.connection.CachingConnectionFactory;
import org.springframework.amqp.rabbit.connection.ConnectionFactory;
import org.springframework.amqp.rabbit.core.RabbitAdmin;
import org.springframework.amqp.rabbit.core.RabbitTemplate;
import org.springframework.amqp.support.converter.Jackson2JsonMessageConverter;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.Primary;
@Slf4j
@Configuration
public class MessageQueueConfiguration {@Value("${spring.rabbitmq.template.reply-timeout:1800000}")private Integer replyTimeout;/*** 存在此名字的bean 自带的容器工厂会不加载(yml下rabbitmq下的template的配置),* 如果想自定义来区分开 需要改变bean 的名称* 配置的其他的bean也都遵循这个规则配置* @param connectionFactory* @return*/@Beanpublic RabbitTemplate rabbitTemplate(ConnectionFactory connectionFactory) {RabbitTemplate template = new RabbitTemplate(connectionFactory);/*** 单位:毫秒* 同步消息方法convertSendAndReceive(),发送端等待接收消费端给出return msg的时间*/template.setReplyTimeout(replyTimeout);template.setMessageConverter(new Jackson2JsonMessageConverter());initMessageSendConfirm(template);return template;}@Beanpublic RabbitAdmin rabbitAdmin(ConnectionFactory connectionFactory) {return new RabbitAdmin(connectionFactory);}@Beanpublic SimpleRabbitListenerContainerFactory rabbitListenerContainerFactory(ConnectionFactory connectionFactory) {SimpleRabbitListenerContainerFactory factory = new SimpleRabbitListenerContainerFactory();factory.setConnectionFactory(connectionFactory);//设置手动ACKfactory.setAcknowledgeMode(AcknowledgeMode.MANUAL);factory.setMessageConverter(new Jackson2JsonMessageConverter());return factory;}@Bean(name = "connectionFactory")@Primarypublic ConnectionFactory connectionFactory(@Value("${spring.rabbitmq.host}") String host,@Value("${spring.rabbitmq.port}") int port,@Value("${spring.rabbitmq.username}") String username,@Value("${spring.rabbitmq.password}") String password) {CachingConnectionFactory connectionFactory = new CachingConnectionFactory();connectionFactory.setHost(host);connectionFactory.setPort(port);connectionFactory.setUsername(username);connectionFactory.setPassword(password);connectionFactory.setVirtualHost("/");connectionFactory.setRequestedHeartBeat(60);/*** CORRELATED:异步回调,消息发送到交换机时会回调这个ConfirmCallback* SIMPLE:则不会出发ConfirmCallback*/connectionFactory.setPublisherConfirmType(CachingConnectionFactory.ConfirmType.CORRELATED);return connectionFactory;}private void initMessageSendConfirm(RabbitTemplate rabbitTemplate) {/*** ConfirmCallback为发送Exchange(交换器)时回调,成功或者失败都会触发;*/rabbitTemplate.setConfirmCallback((correlationData, ack, cause) -> {if (ack) {log.info("消息发送到exchange成功");} else {log.error("消息发送到exchange失败,原因: {}, CorrelationData: {}", cause,correlationData);}});/*** Mandatory为true时,消息通过交换器无法匹配到队列会返回给生产者 并触发ReturnCallback* 为false时,匹配不到会直接被丢弃*//*** Mandatory为true时,消息通过交换器无法匹配到队列会返回给生产者 并触发ReturnCallback* 为false时,匹配不到会直接被丢弃** spring.rabbitmq.template.mandatory属性的优先级高于spring.rabbitmq.publisher-returns的优先级* 一般不设置publisher-returns* spring.rabbitmq.template.mandatory属性可能会返回三种值null、false、true.* spring.rabbitmq.template.mandatory结果为true、false时会忽略掉spring.rabbitmq.publisher-returns属性的值* spring.rabbitmq.template.mandatory结果为null(即不配置)时结果由spring.rabbitmq.publisher-returns确定*/rabbitTemplate.setMandatory(true);/*** ReturnCallback为路由不到队列时触发,成功则不触发;*/rabbitTemplate.setReturnCallback((message, replyCode, replyText, exchange, routingKey) -> {log.error("消息:{} 发送失败, 应答码:{} 原因:{} 交换机: {}  路由键: {}", message,replyCode, replyText,exchange, routingKey);});}
}

2.3.2、设置replyTimeout

单位:毫秒,同步消息,方法convertSendAndReceive(),发送端等待接收消费端给出return msg的时间。
convertSendAndReceive方法是mq的同步方法,调用该方法会阻塞主方法,直到消费端消费完才继续往下走。replyTimeout设置的是消费端执行的最大时间,如果超过设置的时间还没执行完,则会报错。

2.3.3、publisher-confirm-type和mandatory

这两个配置代码注释很详细,可以看注释理解,直接拿来用即可。

2.4、自定义发送消息帮助类MessageQueueHelper

import lombok.extern.slf4j.Slf4j;
import org.springframework.amqp.core.BindingBuilder;
import org.springframework.amqp.core.Queue;
import org.springframework.amqp.core.TopicExchange;
import org.springframework.amqp.rabbit.core.RabbitAdmin;
import org.springframework.amqp.rabbit.core.RabbitTemplate;
import org.springframework.amqp.support.converter.Jackson2JsonMessageConverter;
import org.springframework.stereotype.Component;import javax.annotation.PostConstruct;
import javax.annotation.Resource;
import java.util.UUID;@Slf4j
@Component
public class MessageQueueHelper {@Resourceprivate RabbitTemplate rabbitTemplate;@Resourceprivate RabbitAdmin rabbitAdmin;@PostConstructpublic void init() {rabbitTemplate.setMessageConverter(new Jackson2JsonMessageConverter());}/*** 发送异步消息,根据参数动态创建交换机、队列,和业务更解耦** @param exchangeName* @param queueName* @param sendMessage*/public void sendMessage(String exchangeName, String queueName, String routingKey, Object sendMessage) {try {TopicExchange exchange = new TopicExchange(exchangeName);rabbitAdmin.declareExchange(exchange);Queue queue = new Queue(queueName);rabbitAdmin.declareQueue(queue);String simpleName = sendMessage.getClass().getSimpleName();/*** *(星号)可以代替一个单词* #(井号)可以替代零个或多个单词*/rabbitAdmin.declareBinding(BindingBuilder.bind(queue).to(exchange).with(simpleName.toLowerCase() + ".#"));rabbitTemplate.convertAndSend(exchangeName, routingKey, sendMessage, message -> {/*** 指定消费结果返回的队列*/message.getMessageProperties().setReplyTo("result-stu");message.getMessageProperties().setCorrelationId(UUID.randomUUID().toString());return message;});} catch (Exception e) {e.printStackTrace();}}/*** 发送同步消息** @param exchangeName* @param queueName* @param sendMessage*/public void sendMessageAndReceive(String exchangeName, String queueName, Object sendMessage) {try {TopicExchange exchange = new TopicExchange(exchangeName);rabbitAdmin.declareExchange(exchange);Queue queue = new Queue(queueName);rabbitAdmin.declareQueue(queue);/*** *(星号)可以代替一个单词* #(井号)可以替代零个或多个单词*/String routingKey = "vm.#";rabbitAdmin.declareBinding(BindingBuilder.bind(queue).to(exchange).with(routingKey));Object o = rabbitTemplate.convertSendAndReceive(exchangeName, "vm.fff", sendMessage);System.out.println(o);} catch (Exception e) {e.printStackTrace();}}
}

2.4、消费者

import cn.yx.zg.pojo.Mail;
import cn.yx.zg.pojo.Sms;
import com.rabbitmq.client.Channel;
import lombok.extern.slf4j.Slf4j;
import org.springframework.amqp.core.Message;
import org.springframework.amqp.rabbit.annotation.RabbitHandler;
import org.springframework.amqp.rabbit.annotation.RabbitListener;
import org.springframework.stereotype.Component;@Component
@Slf4j
public class TestConsumer {@RabbitListener(queues = "mail.send")@RabbitHandlerpublic String testConsumer(Mail mail, Channel channel, Message message) throws Exception {log.info("消费消息:{}", mail.toString());/*** ACK,用的最多的一种* deliveryTag:该消息的index* false:表示不是批量*/channel.basicAck(message.getMessageProperties().getDeliveryTag(), false);/*** Nack:手动拒绝* deliveryTag:该消息的index* false:表示不是批量* false:被拒绝的是否重新入队列,一般默认false,因为第一次被拒绝后,后面多次肯定也被拒绝*/
//        channel.basicNack(message.getMessageProperties().getDeliveryTag(), false, false);/*** Reject:手动拒绝,和Nack相比少一个参数* deliveryTag:该消息的index* false:被拒绝的是否重新入队列,一般默认false,因为第一次被拒绝后,后面多次肯定也被拒绝*/
//        channel.basicReject(message.getMessageProperties().getDeliveryTag(), false);return "消费-result";}@RabbitListener(queues = "sms.send")@RabbitHandlerpublic String testConsumer2(Sms sms, Channel channel, Message message) throws Exception {log.info("消费消息:{}", sms.toString());/*** ACK,用的最多的一种* deliveryTag:该消息的index* false:表示不是批量*/channel.basicAck(message.getMessageProperties().getDeliveryTag(), false);/*** Nack:手动拒绝* deliveryTag:该消息的index* false:表示不是批量* false:被拒绝的是否重新入队列,一般默认false,因为第一次被拒绝后,后面多次肯定也被拒绝*/
//        channel.basicNack(message.getMessageProperties().getDeliveryTag(), false, false);/*** Reject:手动拒绝,和Nack相比少一个参数* deliveryTag:该消息的index* false:被拒绝的是否重新入队列,一般默认false,因为第一次被拒绝后,后面多次肯定也被拒绝*/
//        channel.basicReject(message.getMessageProperties().getDeliveryTag(), false);return "消费-result";}

2.5、两个pojo

import lombok.Data;
import lombok.ToString;import java.io.Serializable;/*** 邮件*/
@Data
@ToString
public class Mail implements Serializable {private String mailId;private String content;public Mail() {}public Mail(String mailId, String content) {this.mailId = mailId;this.content = content;}
}
import lombok.Data;
import lombok.ToString;import java.io.Serializable;
@Data
@ToString
public class Sms implements Serializable {private String smsId;private String content;public Sms() {}public Sms(String smsId, String content) {this.smsId = smsId;this.content = content;}}

2.6、Controller

    @Resourceprivate MessageQueueHelper messageQueueHelper;@RequestMapping("send")public void sendMsage() {Mail mail = new Mail("1","我是邮件");messageQueueHelper.sendMessage("message_ex", "mail.send", "mail", mail);Sms sms = new Sms("1","我是短信");messageQueueHelper.sendMessage("message_ex", "sms.send", "sms", sms);}

访问地址:http://localhost:8080/send 测试消息

3、SpringBoot整合RabbitMq,用SimpleMessageListenerContainer更复杂业务的封装

下面封装的这个属于比较复杂的业务,很多公司也是用不到的,有兴趣的可以了解一下。

先说下上面的缺点,其实也不算缺点,比如想让同一个消费者消费多个队列的数据,这样我们就得写多个@RabbitListener(queues = “mail.send”), 这里有个比较高级的写法,就是通过SimpleMessageListenerContainer动态的设置队列的消费类。

还是以上面代码为基础, 先把上面的消费者给注释,我们重新写个消费者。

新建类:TestConsumerListener.java

import cn.yx.zg.pojo.Mail;
import cn.yx.zg.pojo.Sms;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Component;@Component
@Slf4j
public class TestConsumerListener {/*** 注意方法名称,一定要是handleMessage* @param mail* @return* @throws Exception*/public String handleMessage(Mail mail) throws Exception {log.info("消费消息listener:{}", mail.toString());/*** ACK,用的最多的一种* deliveryTag:该消息的index* false:表示不是批量*/
//        channel.basicAck(message.getMessageProperties().getDeliveryTag(), false);/*** Nack:手动拒绝* deliveryTag:该消息的index* false:表示不是批量* false:被拒绝的是否重新入队列,一般默认false,因为第一次被拒绝后,后面多次肯定也被拒绝*/
//        channel.basicNack(message.getMessageProperties().getDeliveryTag(), false, false);/*** Reject:手动拒绝,和Nack相比少一个参数* deliveryTag:该消息的index* false:被拒绝的是否重新入队列,一般默认false,因为第一次被拒绝后,后面多次肯定也被拒绝*/
//        channel.basicReject(message.getMessageProperties().getDeliveryTag(), false);return "消费-result";}public String handleMessage(Sms sms) throws Exception {log.info("消费消息listener:{}", sms.toString());/*** ACK,用的最多的一种* deliveryTag:该消息的index* false:表示不是批量*/
//        channel.basicAck(message.getMessageProperties().getDeliveryTag(), false);/*** Nack:手动拒绝* deliveryTag:该消息的index* false:表示不是批量* false:被拒绝的是否重新入队列,一般默认false,因为第一次被拒绝后,后面多次肯定也被拒绝*/
//        channel.basicNack(message.getMessageProperties().getDeliveryTag(), false, false);/*** Reject:手动拒绝,和Nack相比少一个参数* deliveryTag:该消息的index* false:被拒绝的是否重新入队列,一般默认false,因为第一次被拒绝后,后面多次肯定也被拒绝*/
//        channel.basicReject(message.getMessageProperties().getDeliveryTag(), false);return "消费-result";}
}

修改类:MessageQueueHelper.java

import cn.yx.zg.consumer.TestConsumerListener;
import lombok.extern.slf4j.Slf4j;
import org.springframework.amqp.core.BindingBuilder;
import org.springframework.amqp.core.Queue;
import org.springframework.amqp.core.TopicExchange;
import org.springframework.amqp.rabbit.connection.CachingConnectionFactory;
import org.springframework.amqp.rabbit.core.RabbitAdmin;
import org.springframework.amqp.rabbit.core.RabbitTemplate;
import org.springframework.amqp.rabbit.listener.SimpleMessageListenerContainer;
import org.springframework.amqp.rabbit.listener.adapter.MessageListenerAdapter;
import org.springframework.amqp.support.converter.Jackson2JsonMessageConverter;
import org.springframework.stereotype.Component;import javax.annotation.PostConstruct;
import javax.annotation.Resource;
import java.util.UUID;@Slf4j
@Component
public class MessageQueueHelper {@Resourceprivate RabbitTemplate rabbitTemplate;@Resourceprivate RabbitAdmin rabbitAdmin;@Resourceprivate CachingConnectionFactory cachingConnectionFactory;@Resourceprivate TestConsumerListener testConsumerListener;@PostConstructpublic void init() {rabbitTemplate.setMessageConverter(new Jackson2JsonMessageConverter());startListenerForConsumer(testConsumerListener);}/*** 发送异步消息,根据参数动态创建交换机、队列,和业务更解耦** @param exchangeName* @param queueName* @param sendMessage*/public void sendMessage(String exchangeName, String queueName, String routingKey, Object sendMessage) {try {TopicExchange exchange = new TopicExchange(exchangeName);rabbitAdmin.declareExchange(exchange);Queue queue = new Queue(queueName);rabbitAdmin.declareQueue(queue);String simpleName = sendMessage.getClass().getSimpleName();/*** *(星号)可以代替一个单词* #(井号)可以替代零个或多个单词*/rabbitAdmin.declareBinding(BindingBuilder.bind(queue).to(exchange).with(simpleName.toLowerCase() + ".#"));rabbitTemplate.convertAndSend(exchangeName, routingKey, sendMessage, message -> {/*** 指定消费结果返回的队列*/message.getMessageProperties().setReplyTo("result-stu");message.getMessageProperties().setCorrelationId(UUID.randomUUID().toString());return message;});} catch (Exception e) {e.printStackTrace();}}/*** 发送同步消息** @param exchangeName* @param queueName* @param sendMessage*/public void sendMessageAndReceive(String exchangeName, String queueName, Object sendMessage) {try {TopicExchange exchange = new TopicExchange(exchangeName);rabbitAdmin.declareExchange(exchange);Queue queue = new Queue(queueName);rabbitAdmin.declareQueue(queue);/*** *(星号)可以代替一个单词* #(井号)可以替代零个或多个单词*/String routingKey = "vm.#";rabbitAdmin.declareBinding(BindingBuilder.bind(queue).to(exchange).with(routingKey));Object o = rabbitTemplate.convertSendAndReceive(exchangeName, "vm.fff", sendMessage);System.out.println(o);} catch (Exception e) {e.printStackTrace();}}public void startListenerForConsumer(Object listener) {SimpleMessageListenerContainer simpleMessageListenerContainer = new SimpleMessageListenerContainer(cachingConnectionFactory);MessageListenerAdapter adapter = new MessageListenerAdapter(listener,new Jackson2JsonMessageConverter());simpleMessageListenerContainer.setMessageListener(adapter);//针对哪些队列(参数为可变参数)simpleMessageListenerContainer.setQueueNames("mail.send","sms.send");//同时有多少个消费者线程在消费这个队列,相当于线程池的线程数字。simpleMessageListenerContainer.setConcurrentConsumers(6);//最大的消费者线程数simpleMessageListenerContainer.setMaxConcurrentConsumers(6);/*** 这种设置监听对象的方式,需要重新设置ACK方式,* 不过这里我们设置了手动ACK和MessageListener,并不会触发消费者,就先不设置了,很多业务也不用手动ACK。* 队列消费的结果,还是回放到我们发送消息时设置的返回队列*/
//        //手动确认(单条确认)
//        simpleMessageListenerContainer.setAcknowledgeMode(AcknowledgeMode.MANUAL);
//        simpleMessageListenerContainer.setMessageListener((ChannelAwareMessageListener) (message, channel) -> {
//            log.info("消费端接收到的消息:[{}]", message);
//            channel.basicAck(message.getMessageProperties().getDeliveryTag(), false);
//        });//消费端限流simpleMessageListenerContainer.setPrefetchCount(1);simpleMessageListenerContainer.start();}}

访问地址:http://localhost:8080/send 测试消息

4、死信队列和延迟队列

很简单,这里不举列子了

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

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

相关文章

【C语言】探索数据结构:单链表和双链表

目录 &#x1f4a1;链表的概念和结构 &#x1f4a1;链表的分类 &#x1f4a1;无头单向非循环链表&#xff08;单链表&#xff09;的实现 定义节点结构 单链表的尾部插入 单链表的头部插入 单链表的尾部删除 单链表的头部删除 在指定位置插入前数据 在指定位置之后插入数…

w3c标准

w3c&#xff08;万维网联盟&#xff09;&#xff1a;是一个公益基金组织&#xff0c;由互联网之父&#xff08;博纳斯李&#xff09;牵头组织起来的&#xff0c;用于专门维护互联网相关技术的发展和规范。 w3c标准&#xff0c;不是一个标准&#xff0c;而是一系列标准的集合&am…

人工智能基础-Numpy.array基本操作

基本属性 查看维度 x.ndim查看维度&#xff08;元组形式&#xff09; x.shape元素个数 x.size数据访问 子矩阵 内容同步修改 加是copy&#xff08;&#xff09;则不同步修改 Reshape 修改维度 参数为-1时自动识别个数 合并 np.concatenate([x, y])沿着列合并 np.co…

Springboot使用数据库连接池druid

springboot框架中可以使用druid进行数据库连接池&#xff0c;下面介绍druid在springboot中使用和参数配置介绍。 数据库连接池&#xff08;Druid&#xff09;是一种用于管理数据库连接的机制&#xff0c;其工作原理和常见使用方法如下&#xff1a; 原理&#xff1a;数据库连接…

kafka实现延迟队列

前言 首先说一下延迟队列这个东西&#xff0c;实际上实现他的方法有很多&#xff0c;kafka实现并不是一个最好的选择&#xff0c;例如redis的zset可以实现&#xff0c;rocketmq天然的可以实现&#xff0c;rabbitmq也可以实现。如果切换前几种方案成本高的情况下&#xff0c;那…

OSDI 2023: LVMT: An Efficient Authenticated Storage for Blockchain

我们使用以下6个分类标准对本文的研究选题进行分析: 1. 研究方向: 区块链可扩展性: 提高交易吞吐量和减少确认时间的研究,例如零知识证明、分片和状态通道。密码学技术: 开发或改进用于区块链应用的新密码原语,例如椭圆曲线、承诺方案和累加器。区块链存储和效率: 优化区块…

02神经网络的学习及代码实现

“学习”是指从训练数据中自动获取最优权重参数的过程。引入损失函数指标&#xff0c;学习的目的是以该损失函数为基准&#xff0c;找出尽可能小的损失函数的值。 1、从数据中学习 从数据中学习规律&#xff0c;模式&#xff0c;避免人为介入。 先从图像中提取特征量&#x…

【GAMES101】Lecture 13 光线追踪 Whitted-Style

目录 光线追踪 基本的光线追踪算法 Whitted-Style光线追踪 求曲面交点 求三角形交点 Mller Trumbore Algorithm&#xff08;MT算法&#xff09; 光线追踪 这里讲一下为什么我们需要光线追踪&#xff0c;主要是因为光栅化没有办法很好的处理全局的光照效果&#xff0c;就…

MySQL备份和恢复(二)mysqldump

注意&#xff1a;mysqldump是完全备份 一、mysqldump备份命令 1、 备份数据库 含创建库语句 &#xff08;1&#xff09;备份指定数据库 完全备份一个或多个完整的库&#xff0c; mysqldump -uroot -p[密码] --databases 库名1 [库名2].. >/备份路径/备份文件名.sql#导出…

如何恢复已删除的照片?

在这篇综合文章中发现恢复丢失照片的有效且免费的方法。无论您使用的是智能手机、iPhone、Windows 计算机、Mac、SD 卡还是数码相机&#xff0c;我们都提供有关如何恢复已删除照片的分步说明。此外&#xff0c;学习一些有价值的技巧&#xff0c;以防止将来意外删除照片。 意外…

2024.1.28周报

目录 摘要 ABSTRACT 一、文献阅读 1、题目 2、摘要 3、解决的问题 4、算法模型 5、总结 二、PINN方法 三、PINN神经网络源码 总结 摘要 本周我阅读了一篇题目为Physics Informed Deep Learning (Part I): Data-driven Solutions of Nonlinear Partial Differential…

配置vite自动按需引入 vant 组件

为什么学 按需加载可以减少包体积,优化加载性能 学习内容 全局注册组件 import 需要的组件import 组件样式使用 app.use 注册组件 Tree Shaking 介绍使用 什么是 tree shaking&#xff1f; Tree shaking是一种优化技术&#xff0c;用于减少JavaScript或其他编程语言中未被使用…

【2024程序员必看】鸿蒙应用开发行业分析

鸿蒙操作系统沉浸四年&#xff0c;这次终于迎来了破局的机会&#xff0c;自从2023年华为秋季发布会上宣布鸿蒙 Next操作系统不在兼容Android后&#xff0c;就有不少大厂开始陆续与华为达成了鸿蒙原生应用的开发合作&#xff0c;据1月18日华为官方宣布110多天的产业合力“突进”…

python+selenium自动化测试项目实战

说明&#xff1a;本项目采用流程控制思想&#xff0c;未引用unittest&pytest等单元测试框架 一.项目介绍 目的 测试某官方网站登录功能模块可以正常使用 用例 1.输入格式正确的用户名和正确的密码&#xff0c;验证是否登录成功&#xff1b; 2.输入格式正确的用户名和不…

【C语言】(12)指针

指针在C语言中是一个非常重要的概念&#xff0c;它为程序员提供了直接访问内存的能力&#xff0c;使得数据操作更加灵活高效。理解并正确使用指针是掌握C语言的关键之一。 1. 指针的基本概念 指针本质上是一个变量&#xff0c;其存储的是另一个变量的内存地址。通过指针&…

单例模式有几种写法?请谈谈你的理解?

为什么有单例模式&#xff1f; 单例模式&#xff08;Singleton&#xff09;&#xff0c;也叫单子模式&#xff0c;是一种常用的软件设计模式。在应用这个模式时&#xff0c;单例对象的类必须保证只有一个实例存在。许多时候整个系统只需要拥有一个全局对象&#xff0c;这样有利…

测试用例的书写方式以及测试模板大全

一个优秀的测试用例&#xff0c;应该包含以下信息&#xff1a; 1 &#xff09; 软件或项目的名称 2 &#xff09; 软件或项目的版本&#xff08;内部版本号&#xff09; 3 &#xff09; 功能模块名 4 &#xff09; 测试用例的简单描述&#xff0c;即该用例执行的目的或方法…

SpringMVC实现对网页的访问,在请求控制器中创建处理请求的方法

目录 测试HelloWorld RequestMapping注解 RequestMapping注解的位置 RequestMapping注解的value属性 RequestMapping注解的method属性 SpringMVC支持路径中的占位符&#xff08;重点&#xff09; SpringMVC获取请求参数 1、通过ServletAPI获取 2、通过控制器方法的形参…

头歌C++之while循环性质

目录 第1关:求1到n间所有整数的和 本关必读 本关任务 测试说明 第2关:计算x的n次方 本关必读 本关任务 测试说明 第3关:求给定正整数的“亲密对数” 本关必读 本关任务 测试说明 第4关:判断正整数n的各位数字中是否包含数字3或4

Spring-boot项目+Rancher6.3部署+Nacos配置中心+Rureka注册中心+Harbor镜像仓库+NFS存储

目录 一、项目概述二、环境三、部署流程3.1 Harbor部署3.1.1 docker安装3.1.2 docker-compose安装3.1.3 安装证书3.1.4 Harbor下载配置安装 3.2 NFS存储搭建3.3 Rancher平台配置3.3.1 NFS存储相关配置3.3.2 Harbor相关配置3.3.3 Nacos部署及相关配置3.3.4 工作负载deployment配…