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

构建稳定高效的消息传递中间件:消息队列系统的设计与实现



✨✨谢谢大家捧场,祝屏幕前的小伙伴们每天都有好运相伴左右,一定要天天开心哦!✨✨ 
🎈🎈作者主页: 喔的嘛呀🎈🎈

目录

一、引言

二、设计目标

2.1、高可用性

1. 集群搭建

1.1 Docker Compose配置示例

2. 容错和恢复

2.1 异常检测与自动故障迁移

2.2 Spring Boot代码示例

3. 心跳检测

4. 优雅关闭

2.2、持久性

1. 生产者消息持久化

2. 队列的持久化

3. 消息队列的持久化

总结

2.3、低延迟

1. 异步消息传递

2. 批量消息发送

3. 消息序列化优化

4. 配置消息队列优化参数

总结

2.4、可伸缩性

1. 集群搭建

Docker Compose配置示例

2. 消费者的水平扩展

3. 动态调整资源配置

总结

2.5 顺序性

1. 单一队列实现顺序性

2. 消费者的单线程处理

3. 利用消息的消息头或属性

总结

2.6、监控与管理

1. 健康检查

2. 监控指标

3. 日志和错误处理

4. 运维工具

5. 定期维护

总结

三、架构设计

1. 组件设计

1.1 生产者

1.2 消费者

1.3 消息队列服务

1.4 管理与监控组件

2. 数据存储

3. 分布式一致性

总结


一、引言

消息队列是分布式系统中至关重要的组件之一,用于实现异步通信、解耦系统模块、缓解系统压力等。在本文中,我们将深入讨论如何设计一个高效可靠的消息队列系统,以满足各种应用场景的需求。

二、设计目标

在开始设计之前,我们首先要明确设计目标。一个理想的消息队列系统应该具备以下特性:

  1. 高可用性:系统应该能够容忍节点故障,保持高可用性。
  2. 持久性:消息在传递过程中不应该丢失,即使系统发生故障。
  3. 低延迟:消息传递应该保持低延迟,以满足实时性要求。
  4. 可伸缩性:系统应该能够处理不断增长的负载,支持水平扩展。
  5. 顺序性:对于需要保持顺序的消息,系统应该能够提供有序性保证。
  6. 监控与管理:提供有效的监控和管理手段,便于系统运维。

2.1、高可用性

1. 集群搭建

搭建RabbitMQ集群以确保高可用性。一个RabbitMQ集群由多个节点组成,每个节点分布在不同的物理服务器上。在节点之间进行数据同步,保证即使某个节点宕机,其他节点仍能提供服务。

1.1 Docker Compose配置示例
version: '3.7'
services:rabbitmq-node1:image: "rabbitmq:3.8-management"container_name: "rabbitmq-node1"ports:- "5672:5672"- "15672:15672"networks:- rabbitmq-networkenvironment:RABBITMQ_ERLANG_COOKIE: "secret_cookie"RABBITMQ_NODENAME: "rabbit@node1"RABBITMQ_CLUSTER_NODE_NAME: "rabbit@node1"RABBITMQ_CLUSTER_NAME: "rabbit-cluster"restart: alwaysrabbitmq-node2:image: "rabbitmq:3.8-management"container_name: "rabbitmq-node2"ports:- "5673:5672"- "15673:15672"networks:- rabbitmq-networkenvironment:RABBITMQ_ERLANG_COOKIE: "secret_cookie"RABBITMQ_NODENAME: "rabbit@node2"RABBITMQ_CLUSTER_NODE_NAME: "rabbit@node2"RABBITMQ_CLUSTER_NAME: "rabbit-cluster"restart: alwaysnetworks:rabbitmq-network:

这是一个简单的Docker Compose配置,其中包含两个RabbitMQ节点,它们通过Erlang Cookie和集群名称连接到一起。

2. 容错和恢复

2.1 异常检测与自动故障迁移

在RabbitMQ中,可以使用 RabbitMQ Federation 插件来设置镜像队列,确保队列中的消息被复制到多个节点。如果某个节点发生故障,系统可以自动从其他节点获取数据,保证消息的可用性。

2.2 Spring Boot代码示例
@Configuration
public class RabbitMQConfig {@Beanpublic CachingConnectionFactory cachingConnectionFactory() {CachingConnectionFactory connectionFactory = new CachingConnectionFactory();connectionFactory.setAddresses("node1:5672,node2:5672");connectionFactory.setUsername("guest");connectionFactory.setPassword("guest");connectionFactory.setVirtualHost("/");return connectionFactory;}@Beanpublic RabbitTemplate rabbitTemplate(ConnectionFactory connectionFactory) {RabbitTemplate rabbitTemplate = new RabbitTemplate(connectionFactory);rabbitTemplate.setMessageConverter(new Jackson2JsonMessageConverter());return rabbitTemplate;}
}

在Spring Boot中,配置 CachingConnectionFactory 来指定RabbitMQ节点的地址。通过设置多个节点的地址,Spring Boot将在节点之间建立连接,以实现高可用性。

3. 心跳检测

使用RabbitMQ提供的心跳检测机制,监测节点的健康状态。如果某个节点无响应,系统可以自动将其从集群中排除,并启用备用节点。

4. 优雅关闭

在系统关闭或者节点下线时,确保消息队列系统的优雅关闭。Spring Boot提供 @PreDestroy 注解,可以用于执行一些清理工作。

@Component
public class ShutdownHook {@PreDestroypublic void onShutdown() {// 执行关闭前的清理工作,如释放资源等System.out.println("Shutting down gracefully...");}
}

通过搭建RabbitMQ集群、设置镜像队列、使用心跳检测和优雅关闭机制,我们可以构建一个具有高可用性的消息队列系统。在实际应用中,还需要根据业务场景和系统需求进行更详细的配置和调整,确保系统稳定可靠地运行。

2.2、持久性

确保消息的持久性是构建可靠消息队列系统的关键要素。在消息队列中,持久性通常涉及到两个方面:生产者将消息持久化到队列,以及消息队列本身的持久化。

以下是实现消息持久性的关键步骤和Java代码示例,使用RabbitMQ作为消息队列系统。

1. 生产者消息持久化

当生产者将消息发送到队列时,可以设置消息的持久性属性。这样,即使在消息被发送到队列但还未被处理时,系统发生故障,消息也不会丢失。

import org.springframework.amqp.core.AmqpTemplate;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;@Component
public class MessageProducer {@Autowiredprivate AmqpTemplate amqpTemplate;public void sendMessage(String message) {// 设置消息持久性amqpTemplate.convertAndSend("exchange", "routingKey", message, messagePostProcessor -> {messagePostProcessor.getMessageProperties().setDeliveryMode(MessageDeliveryMode.PERSISTENT);return messagePostProcessor;});}
}

在上述示例中,setDeliveryMode(MessageDeliveryMode.PERSISTENT) 将消息标记为持久性。这样,即使RabbitMQ节点在消息进入队列但尚未被消费者处理时发生故障,消息也会在节点重新启动后仍然存在。

2. 队列的持久化

为了确保即使RabbitMQ服务器重启,队列中的消息也不会丢失,我们需要将队列设置为持久化。

import org.springframework.amqp.core.Queue;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;@Configuration
public class QueueConfiguration {@Beanpublic Queue exampleQueue() {// 设置队列持久性return new Queue("exampleQueue", true);}
}

在这个示例中,通过将 Queue 构造函数的第二个参数设置为 true,我们将队列标记为持久化的。这意味着队列的元数据和消息将在RabbitMQ服务器重启后继续存在。

3. 消息队列的持久化

确保RabbitMQ服务器本身的持久性也是重要的。通过在RabbitMQ的配置文件中设置 disk_free_limit 等参数,可以将消息队列的元数据和消息数据保存在持久化存储中,以防止数据丢失。

总结

通过设置生产者发送的消息和队列本身的持久性,以及确保消息队列系统本身的持久性,我们可以在系统发生故障时保证消息不会丢失。这对于构建可靠、稳定的消息队列系统至关重要,特别是在面对复杂分布式系统的挑战时。

2.3、低延迟

实现低延迟的消息传递对于满足实时性的要求至关重要。在构建消息队列系统时,我们可以采取一些策略和技术来降低消息传递的延迟。以下是一些关键点和Java代码示例,以使用RabbitMQ实现低延迟的消息传递。

1. 异步消息传递

使用异步消息传递机制,允许生产者发送消息而无需等待消费者的响应。这样可以在消息发送的同时继续进行其他操作,提高系统的并发性和响应速度。

import org.springframework.amqp.rabbit.core.RabbitTemplate;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;@Component
public class AsyncMessageProducer {@Autowiredprivate RabbitTemplate rabbitTemplate;public void sendAsyncMessage(String message) {// 发送异步消息rabbitTemplate.convertAndSend("exchange", "routingKey", message);// 继续执行其他操作,而不必等待消息被消费}
}

在这个示例中,convertAndSend 是一个非阻塞的方法,它将消息发送到队列而不会等待确认。

2. 批量消息发送

将多个消息打包成一个批量发送,可以降低每个消息的传递时间。这对于需要频繁发送小量消息的场景特别有效。

import org.springframework.amqp.rabbit.core.RabbitTemplate;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;@Component
public class BatchMessageProducer {@Autowiredprivate RabbitTemplate rabbitTemplate;public void sendBatchMessage(List<String> messages) {// 批量发送消息for (String message : messages) {rabbitTemplate.convertAndSend("exchange", "routingKey", message);}}
}

在这个示例中,通过循环将多个消息发送到队列,可以减少每个消息的传递开销。

3. 消息序列化优化

选择高效的消息序列化方式,以降低消息的传递时间。例如,使用二进制序列化方式(如Google Protocol Buffers或MessagePack)相比于JSON或XML通常具有更高的性能。

import com.fasterxml.jackson.databind.ObjectMapper;
import org.springframework.amqp.rabbit.core.RabbitTemplate;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;@Component
public class BinarySerializationMessageProducer {@Autowiredprivate RabbitTemplate rabbitTemplate;@Autowiredprivate ObjectMapper objectMapper;public void sendBinaryMessage(Object message) {// 使用二进制序列化方式发送消息byte[] serializedMessage = objectMapper.writeValueAsBytes(message);rabbitTemplate.convertAndSend("exchange", "routingKey", serializedMessage);}
}

在这个示例中,ObjectMapper 将对象序列化为字节数组,以便使用二进制格式发送。

4. 配置消息队列优化参数

通过配置消息队列的一些优化参数,可以降低传递消息的延迟。例如,设置队列的 x-message-ttl 参数,以定义消息的最大存活时间。

import org.springframework.amqp.core.Queue;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;@Configuration
public class QueueConfiguration {@Beanpublic Queue exampleQueue() {// 设置队列的消息存活时间(毫秒)return new Queue("exampleQueue", true, false, false, ImmutableMap.of("x-message-ttl", 5000));}
}

在这个示例中,通过 ImmutableMap.of("x-message-ttl", 5000) 设置消息的最大存活时间为5秒。

总结

通过采用异步消息传递、批量消息发送、选择高效的消息序列化方式以及配置消息队列的优化参数,我们可以有效降低消息传递的延迟,满足实时性的要求。在实际应用中,需要根据具体场景和需求进行更详细的优化和调整。

2.4、可伸缩性

实现消息队列系统的可伸缩性是确保系统能够处理不断增长负载的关键。以下是一些关键点和Java代码示例,以使用RabbitMQ实现可伸缩的消息队列系统。

1. 集群搭建

构建消息队列系统的集群是实现可伸缩性的基础。通过在不同的节点上部署多个消息队列服务,可以有效地分摊负载并提高系统的整体吞吐量。

Docker Compose配置示例

version: '3.7'
services:rabbitmq-node1:image: "rabbitmq:3.8-management"container_name: "rabbitmq-node1"ports:- "5672:5672"- "15672:15672"networks:- rabbitmq-networkenvironment:RABBITMQ_ERLANG_COOKIE: "secret_cookie"RABBITMQ_NODENAME: "rabbit@node1"RABBITMQ_CLUSTER_NODE_NAME: "rabbit@node1"RABBITMQ_CLUSTER_NAME: "rabbit-cluster"restart: alwaysrabbitmq-node2:image: "rabbitmq:3.8-management"container_name: "rabbitmq-node2"ports:- "5673:5672"- "15673:15672"networks:- rabbitmq-networkenvironment:RABBITMQ_ERLANG_COOKIE: "secret_cookie"RABBITMQ_NODENAME: "rabbit@node2"RABBITMQ_CLUSTER_NODE_NAME: "rabbit@node2"RABBITMQ_CLUSTER_NAME: "rabbit-cluster"restart: alwaysnetworks:rabbitmq-network:

通过在不同的节点上运行RabbitMQ实例,可以形成一个集群,以提高系统的可伸缩性。

2. 消费者的水平扩展

通过将消费者部署在多个实例上,实现对消费者的水平扩展。这样可以提高消息处理的并发性能。

import org.springframework.amqp.rabbit.annotation.RabbitListener;
import org.springframework.stereotype.Component;@Component
public class Consumer {@RabbitListener(queues = "exampleQueue")public void handleMessage(String message) {// 处理接收到的消息System.out.println("Received message: " + message);}
}

在上述示例中,可以启动多个具有相同 @RabbitListener 注解的消费者实例,它们将同时监听同一个队列,实现消费者的水平扩展。

3. 动态调整资源配置

通过动态调整消息队列服务的资源配置,可以根据系统负载实时进行扩展或缩小。在Docker Compose配置中,可以使用scale指令启动多个节点实例。

services:rabbitmq-node1:# ...deploy:replicas: 3  # 指定节点数量# ...rabbitmq-node2:# ...deploy:replicas: 3  # 指定节点数量# ...

在实际应用中,根据系统监控数据和负载情况,动态调整实例数量和配置,以实现系统的弹性伸缩。

总结

通过构建消息队列系统的集群、实现消费者的水平扩展、以及动态调整资源配置,可以有效地提高系统的可伸缩性,使其能够处理不断增长的负载。在实际应用中,需要结合具体场景和需求,根据监控数据进行及时的调整和优化。

2.5 顺序性

保证消息的顺序性是在某些场景中非常重要的需求,例如在处理事务性消息或者处理业务逻辑依赖于消息顺序的情况。在消息队列系统中,有一些策略和技术可以用来确保消息的有序性。

以下是一些关键点和Java代码示例,以使用RabbitMQ实现有序性保证的消息队列系统。

1. 单一队列实现顺序性

将所有需要保持顺序的消息发送到同一个队列,通过单一队列来保证消息的顺序性。

import org.springframework.amqp.rabbit.core.RabbitTemplate;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;@Component
public class OrderedMessageProducer {@Autowiredprivate RabbitTemplate rabbitTemplate;public void sendOrderedMessage(String message, int sequence) {// 发送有序消息,使用sequence作为routing keyrabbitTemplate.convertAndSend("ordered_exchange", String.valueOf(sequence), message);}
}

在上述示例中,使用sequence作为消息的routing key,确保消息发送到同一个队列中。

2. 消费者的单线程处理

在消费者端,可以通过使用单线程来处理消息,确保消费者按照顺序处理消息。

import org.springframework.amqp.rabbit.annotation.RabbitListener;
import org.springframework.stereotype.Component;@Component
public class OrderedConsumer {@RabbitListener(queues = "ordered_queue")public void handleMessage(String message) {// 处理接收到的有序消息System.out.println("Received ordered message: " + message);}
}

在上述示例中,通过@RabbitListener注解,确保消息在同一个消费者实例的单线程中按照顺序处理。

3. 利用消息的消息头或属性

消息队列系统通常提供一些额外的消息头或属性来帮助实现有序性。例如,RabbitMQ提供了x-death属性,可以用于获取消息的死信信息,以此判断消息的处理状态。

import org.springframework.amqp.core.Message;
import org.springframework.amqp.rabbit.annotation.RabbitListener;
import org.springframework.stereotype.Component;@Component
public class OrderedConsumer {@RabbitListener(queues = "ordered_queue")public void handleMessage(Message message) {// 获取消息的死信信息Map<String, Object> deathHeaders = message.getMessageProperties().getHeaders();// 根据死信信息判断消息的处理状态// ...}
}

在实际应用中,可以根据具体的需求和场景选择合适的方式来实现有序性保证。

总结

通过将有序消息发送到同一个队列,使用单线程处理消费者,或者利用消息的消息头或属性,我们可以在消息队列系统中实现有序性保证。在选择策略时,需要根据实际需求和系统规模综合考虑。

2.6、监控与管理

在构建消息队列系统时,提供有效的监控和管理手段对于系统的运维至关重要。这涉及到监控系统的健康状况、性能指标、以及执行维护和管理操作的能力。以下是一些建议和Java代码示例,使用RabbitMQ为例,来实现监控与管理功能。

1. 健康检查

实现一个简单的健康检查接口,供运维人员使用,用于检查消息队列系统是否正常运行。

import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;@RestController
public class HealthCheckController {@GetMapping("/health")public String healthCheck() {// 检查消息队列系统的健康状况// ...return "OK";}
}

在上述示例中,/health 路径是一个用于健康检查的端点,返回 "OK" 表示系统正常。

2. 监控指标

使用监控工具,例如Prometheus或者Spring Boot Actuator,来收集和暴露消息队列系统的性能指标。

import org.springframework.boot.actuate.endpoint.annotation.Endpoint;
import org.springframework.boot.actuate.endpoint.annotation.ReadOperation;
import org.springframework.stereotype.Component;@Component
@Endpoint(id = "rabbitmq")
public class RabbitMQMetricsEndpoint {@ReadOperationpublic RabbitMQMetrics getMetrics() {// 获取RabbitMQ相关的性能指标// ...return new RabbitMQMetrics(/* 指标数据 */);}
}

在上述示例中,通过创建一个自定义的Actuator端点,可以将RabbitMQ的性能指标暴露出来,方便监控。

3. 日志和错误处理

配置合适的日志记录,确保错误和异常情况能够被及时捕获和记录。使用工具或系统来集中收集日志信息,以便于分析和排查问题。

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.amqp.rabbit.listener.ConditionalRejectingErrorHandler;public class CustomErrorHandler extends ConditionalRejectingErrorHandler {private static final Logger logger = LoggerFactory.getLogger(CustomErrorHandler.class);@Overridepublic void handleError(Throwable t) {// 记录错误日志logger.error("Error in RabbitMQ message processing", t);super.handleError(t);}
}

在上述示例中,通过自定义错误处理器,可以在消息处理发生错误时记录相关的日志信息。

4. 运维工具

使用专业的运维工具,例如RabbitMQ Management Plugin,可以通过Web界面轻松地监控和管理RabbitMQ。通过提供友好的用户界面,运维人员可以更方便地查看队列状态、执行操作等。

5. 定期维护

定期执行系统维护操作,例如清理过期的消息、检查节点健康状态、更新系统配置等。可以通过定时任务或者集成现有的调度工具来执行这些操作。

import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Component;@Component
public class MaintenanceTask {@Scheduled(cron = "0 0 2 * * ?") // 每天凌晨2点执行public void performMaintenance() {// 执行系统维护操作// ...}
}

在上述示例中,通过使用Spring的定时任务,可以定期执行维护操作。

总结

通过实现健康检查、暴露监控指标、配置日志和错误处理、使用运维工具以及定期维护等手段,可以为消息队列系统提供有效的监控和管理手段,以便于系统运维。在实际应用中,根据具体需求选择合适的工具和策略。

三、架构设计

1. 组件设计

1.1 生产者

生产者负责将消息发送到消息队列系统中。在设计中,生产者可以通过REST API或者消息协议(如AMQP、Kafka协议)与消息队列进行通信。

1.2 消费者

消费者从消息队列中订阅并消费消息。消费者可以通过订阅特定的主题或队列,以接收感兴趣的消息。

1.3 消息队列服务

消息队列服务是核心组件,负责存储和分发消息。它包含多个节点以确保高可用性,并提供消息的持久性保证。

1.4 管理与监控组件

为了方便系统运维,设计一个管理与监控组件,用于监视消息队列的状态、执行维护操作,并提供合适的API供运维人员使用。

2. 数据存储

消息队列的数据存储通常采用高性能的存储引擎,支持快速读写操作。对于持久性要求高的系统,可以使用分布式文件系统或者分布式数据库。

3. 分布式一致性

保证分布式系统的一致性是设计中的难点之一。采用分布式事务、选举算法和副本机制来确保系统在节点故障时依然保持一致性。

总结

设计一个高效可靠的消息队列系统需要综合考虑架构、一致性、性能和可维护性等多个方面。上述代码示例使用了Spring Boot和RabbitMQ,但具体的技术选型可以根据项目需求和团队熟悉度来定。在实际应用中,还需考虑安全性、容灾性、监控等方面的问题,以确保系统能够稳定、可靠地运行。

相关文章:

构建稳定高效的消息传递中间件:消息队列系统的设计与实现

✨✨谢谢大家捧场&#xff0c;祝屏幕前的小伙伴们每天都有好运相伴左右&#xff0c;一定要天天开心哦&#xff01;✨✨ &#x1f388;&#x1f388;作者主页&#xff1a; 喔的嘛呀&#x1f388;&#x1f388; 目录 一、引言 二、设计目标 2.1、高可用性 1. 集群搭建 1.1 …...

支持 MKV、MP4、AVI、MPG 等格式视频转码器

一、简介 1、一款开源的视频转码器&#xff0c;适用于 Linux、Mac 和 Windows。它是一个免费的工具&#xff0c;由志愿者们开发&#xff0c;可以将几乎所有格式的视频转换为现代、广泛支持的编码格式。你可以在官网上下载该应用或源代码。该软件支持 MKV、MP4、AVI、MPG 等格式…...

yum

文章目录 本地源配置本地yum源仓库yum常用的操作命令 网络源阿里云当yum 安装源代码软件包需要编译安装&#xff0c;需要安装支持c和c程序语言的编译器&#xff0c;如gcc、gcc-c、make 如果使用rpm方式安装&#xff0c;则需要先安装多个依赖包&#xff0c;这样会很繁琐。可以使…...

【单片机毕业设计选题24016】-基于STM32和阿里云的采空区环境监测系统设计

系统功能: 系统分为主机端和从机端&#xff0c;主机端主动向从机端发送信息和命令&#xff0c;从机端 收到主机端的信息后回复温度,甲烷&#xff0c;一氧化碳&#xff0c;氧气和系统状态等信息。 同时主机端将这些信息上传至阿里云服务器。 主要功能模块原理图: 电源时钟烧…...

Leetcode3179. K 秒后第 N 个元素的值

Every day a Leetcode 题目来源&#xff1a;3179. K 秒后第 N 个元素的值 解法1&#xff1a;模拟 模拟 k 轮&#xff0c;数组保存上一次结果&#xff0c;然后计算当前轮次的结果。 代码&#xff1a; /** lc appleetcode.cn id3179 langcpp** [3179] K 秒后第 N 个元素的值…...

vue3第二阶段的开发文档

1 2.1 案例——学习计划表 2.1.1 准备工作 在开发“学习计划表”案例之前&#xff0c;需要先完成一些准备工作&#xff0c;具体步骤如下。 ① 打开命令提示符&#xff0c;切换到 D:\vue\chapter02 目录&#xff0c;在该目录下执行如下命令&#xff0c;创建 项目。 np…...

【网络安全学习】漏洞扫描:- 02- nmap漏洞扫描

1.nmap的介绍 Nmap是一款功能强大的网络探测和安全扫描工具&#xff0c;可以对目标进行端口扫描、服务探测、操作系统指纹识别等操作。 Nmap自带了许多内置的NSE脚本&#xff0c;它们可以根据不同的目标和场景来执行不同的功能。这些脚本存放在Nmap安装目录**/usr/share/nmap…...

Web开发技能树-HTML-class/id/name/tag

1 需求 需求1&#xff1a;CSS查找HTML元素 *tagclassid派生选择器 需求2&#xff1a;JavaScript查找HTML元素 通过id找到HTML元素&#xff1a;document.getElementById()通过标签名找到HTML元素&#xff1a;getElementsByTagName()通过类名找到HTML元素:document.getElemen…...

据APO Research(阿谱尔)统计,2023年全球乳酸企业产能约119.3万吨

乳酸又称 2-羟基丙酸&#xff0c;一种天然有机酸&#xff0c;分子式是 C3H6O3。是自然界中最为广泛存在的羟基酸&#xff0c;于 1780 年被瑞典科学家 Scheele 首次发现。乳酸是自然界最小的手性分子&#xff0c;以两种立体异构体的形式存在于自然界中&#xff0c;即左旋型 L-乳…...

百度文心智能体平台(想象即现实):轻松上手,开启智能新时代!创建属于自己的智能体应用。

目录 1.1、文心智能体平台 1.2、创建智能体 1.3、智能体报名入口 1.4、古诗词小助手 1.5、访问我的智能体 在这个全新的时代里&#xff0c;人工智能技术正以前所未有的速度发展&#xff0c;渗透到我们生活的方方面面。无论是智能家居、自动驾驶&#xff0c;还是医疗诊断、…...

Linux中ls -lsa 和ls -lst区别

在Linux中&#xff0c;ls 命令用于列出目录内容。当与不同的选项组合时&#xff0c;它可以以不同的方式显示文件和目录的详细信息。 对于 ls -lsa 和 ls -lst&#xff0c;它们的主要区别在于显示的列和排序方式&#xff1a; ls -lsa: -l: 使用长格式显示文件和目录的详细信息。…...

TDengine 签约上海晶澳太阳能,助力储能业务平台搭建

在全球能源结构转型和碳中和目标的大背景下&#xff0c;太阳能作为清洁能源的重要组成部分&#xff0c;正逐渐成为新能源发展的关键。作为一个领先的数据处理平台&#xff0c;TDengine 最近与太阳能行业的领头羊晶澳太阳能科技股份有限公司开展了深度合作。这项合作旨在利用 TD…...

【数据结构】选择题

在数据结构中&#xff0c;从逻辑上可以把数据结构分为&#xff08;线性结构和非线性结构&#xff09; 当输入规模为n时&#xff0c;下列算法渐进复杂性中最低的是&#xff08;&#xff09; 时间复杂度 某线性表采用顺序存储结构&#xff0c;每个元素占4个存储单元&#xf…...

数据库 |试卷八试卷九试卷十

1.基数是指元组的个数 2.游标机制 3.触发器自动调用 4.count(*)统计所有行&#xff0c;不忽略空值null&#xff0c;但不但要全局扫描&#xff0c;也要对表的每个字段进行扫描&#xff1b; 5.eacherNO INT NOT NULL UNIQUE&#xff0c;为什么不能断定TeacherNO是主码&#xff…...

【华为HCIA数通网络工程师真题-构建互联互通的IP网络】

文章目录 一、选择题 一、选择题 1、缺省情况下&#xff0c;广播网络上OSPF协议RouterDeadInterval是&#xff1f; 40s &#xff08;ospf 的 RouterDeadInterval 为四倍 hello time 时间&#xff0c;hello time 周期默认为10s&#xff0c;所以 RouterDeadInterval 默认为 40s …...

Kafka 负载均衡挑战及解决思路

本文转载自 Agoda Engineering&#xff0c;介绍了在实际应用中&#xff0c;如何应对 Kafka 负载均衡所遇到的各种挑战&#xff0c;并提出相应的解决思路。本文简要阐述了 Kafka 的并行性机制、常用的分区策略以及在实际操作中遇到的异构硬件、不均匀工作负载等问题。通过深入分…...

【Java面试】二十一、JVM篇(中):垃圾回收相关

文章目录 1、类加载器1.1 什么是类加载器1.2 什么是双亲委派机制 2、类装载的执行过程&#xff08;类的生命周期&#xff09;3、对象什么时候可以被垃圾回收器处理4、JVM垃圾回收算法4.1 标记清除算法4.2 标记整理算法4.3 复制算法 5、分代收集算法5.1 MinorGC、Mixed GC、Full…...

深入理解预处理

1.预定义符号 C语言设置了⼀些预定义符号&#xff0c;可以直接使用&#xff0c;预定义符号也是在预处理期间处理的。 __FILE__ //进⾏编译的源⽂件 __LINE__ //⽂件当前的⾏号 __DATE__ //⽂件被编译的⽇期 __TIME__ //⽂件被编译的时间 __STDC__ //如果编译器遵循ANSI C&…...

DSP28335:定时器

1.定时器介绍 1.1 定时器工作原理 TMS320F28335的CPU Time有三个&#xff0c;分别为Timer0&#xff0c;Timer1&#xff0c;Timer2&#xff0c;其中Timer2是为操作系统DSP/BIOS保留的&#xff0c;当未移植操作系统时&#xff0c;可用来做普通的定时器。这三个定时器的中断信号分…...

系统架构理解

一、统一提前查好所有数据后续逻辑用到啥取啥&#xff0c;还是等用到对应数据的时候再查 1、用到啥查啥&#xff1a; 优势&#xff1a;减少依赖调用次数&#xff0c;减轻服务器压力&#xff1b;代码逻辑清晰&#xff0c;没有太多分支判断 劣势&#xff1a;无法避免串行调用&am…...

uni-app页面的跳转三种方式,功能作用有什么区别?

一、三种方式的作用 1、uni.reLaunch 作用是关闭所有页面&#xff0c;然后打开新的页面 类似于重新启动应用&#xff0c;打开的页面栈会被清空&#xff0c;只显示新打开的页面。使用uni.reLaunch方法可以实现整个应用的重定向 uni.reLaunch({url: /pages/login/login }) 2、…...

React 通信:深层传递(Props、Context、Children Jsx)

在之前的文章 探讨&#xff1a;围绕 props 阐述 React 通信 中总结了关于“父子”组件传值&#xff0c;但是当需要在组件树中深层传递参数以及需要在组件间复用相同的参数时&#xff0c;传递 props 就会变得很麻烦。 实际案例&#xff1a; 下述展示有两种状态&#xff1a;① 详…...

《Windows API每日一练》5.1 键盘基础

本节我们讲述关于键盘的一些基础知识。当我们按下一个键盘按键时&#xff0c;会产生一个键盘按键消息。这一点你能确定吗&#xff1f;假如是一个菜单快捷键消息&#xff0c;或者是一个子窗口控件消息呢&#xff1f;这就超出了本节讨论的范围&#xff0c;我们将在菜单和子窗口控…...

Class.forName()方法总结

Class.forName()方法总结 大家好&#xff0c;我是免费搭建查券返利机器人省钱赚佣金就用微赚淘客系统3.0的小编&#xff0c;也是冬天不穿秋裤&#xff0c;天冷也要风度的程序猿&#xff01;Class.forName()方法是Java反射机制中的一个重要方法&#xff0c;它用于动态加载类并返…...

Python | Leetcode Python题解之第168题Excel表列名称

题目&#xff1a; 题解&#xff1a; class Solution:def convertToTitle(self, columnNumber: int) -> str:ans list()while columnNumber > 0:columnNumber - 1ans.append(chr(columnNumber % 26 ord("A")))columnNumber // 26return "".join(an…...

【ARMv8/ARMv9 硬件加速系列 3.5.2 -- SVE 向量寄存器 有多少位数?】

文章目录 SVE 向量寄存器SVE 向量寄存器大小SVE 可伸缩性的好处SVE 寄存器长度示例SVE 向量寄存器 在 ARMv9 架构中,包括其 Scalable Vector Extension (SVE) 和 Scalable Vector Extension 2 (SVE2) 的增强,向量寄存器(通常称为 Z 寄存器)的大小设计为可伸缩的,以便在不…...

Vulkan入门系列2- 绘制三角形(未完待续)

概述&#xff1a; Vulkan的学习曲线是比较陡峭的&#xff0c;学习Vulkan刚开始像是在爬一个陡坡&#xff0c;等上了这个陡坡之后&#xff0c;后面学习曲线就相对比较平缓了。那么在Vulkan中绘制一个三角形&#xff0c;就相当于是在爬这样一个陡坡&#xff0c;因为绘制三角形需…...

企业UDP文件传输工具测速的方式(下)

在前一篇文章中&#xff0c;我们深入讨论了UDP传输的基本概念和镭速UDP文件传输工具如何使用命令行快速进行速度测试。现在&#xff0c;让我们进一步探索更为高级和灵活的方法&#xff0c;即通过整合镭速UDP的动态或静态库来实现网络速度的测量&#xff0c;以及如何利用这一过程…...

Artalk-CORS,跨域拦截问题

今天重新部署Artalk之后&#xff0c;遇到了CORS——跨域拦截的问题&#xff0c;卡了好一会记录一下。 起因 重新部署之后&#xff0c;浏览器一直提示CORS&#xff0c;之前在其他项目也遇到过类似的问题&#xff0c;原因就在于跨域问题。...

SSL证书怎样配置部署更安全?

在互联网上&#xff0c;SSL证书是用于加密网站与用户之间传输的数据的一种数字证书。它通过建立安全的连接&#xff0c;确保网站的身份和保护用户的隐私&#xff0c;是网站安全的重要组成部分。然而&#xff0c;要想让SSL证书发挥最大的作用&#xff0c;除了检查证书是否过期外…...

济南品牌网站建设介绍/四川seo技术培训

上一节讲了用xml文件的方式配置Bean&#xff0c;虽然可以满足所有要求&#xff0c;但是简单的两个类就配置了那么多内容。后期维护起来很不方便。这一节学习通过注解的方式实现Bean的配置。这里先了解下各个注解代表的含义 controller 控制器&#xff08;注入服务&#xff09; …...

网站卖了对方做违法吗/江苏疫情最新消息

文章目录redis的事务概念redis事务的三个阶段redis事务的四大特性&#xff08;ACID&#xff09;redis中的watch命令了解吗&#xff1f;总结redis的事务概念 Redis通过MULTI、EXEC、WATCH等命令来实现事务( transaction)功能。事务提供了一种将多个命令请求打包&#xff0c;然后…...

济南网站建设模板/蚂蚁bt

题目&#xff1a; 输入两个单调递增的链表&#xff0c;输出两个链表合成后的链表&#xff0c;当然我们需要合成后的链表满足单调不减规则。 代码思路&#xff1a; 两种解法&#xff1a;递归和非递归 解题代码&#xff1a; /* public class ListNode {int val;ListNode next nu…...

网站备案 营业执照副本/什么是营销型网站?

定时器放在了窗体的构造函数中&#xff0c;导致运行崩溃&#xff0c;正确的方法是将定时器放在窗体的OnInitDialog函数中 对于这种错误&#xff0c;一般是窗体未加载完成&#xff0c;使用定时器&#xff0c;或者使用getdlgitem对窗体控件进行操作导致的。解决办法是将相关代码放…...

视频解析网站制作/网络营销推广技巧

MAC帧为CMC与CM之间MAC子层的传输单位。 在上行方向&#xff0c;MAC帧前面为PMD子层开销。 在下行方向&#xff0c;MAC帧前面为MPEG传输汇聚头。 MAC帧&#xff1a;MAC HeaderData PDU MAC帧的开始部分为MAC头&#xff1a;FC(1)&#xff0c;MAC_PARM(1), LEN(2), EHDR(0-240), …...

网上帮人做网站/google国外入口

SMT人才网一套完整的PC需要很多个部件组成&#xff0c;如果在某次灾难中你只能保留一个部件&#xff0c;大家会选择留下哪个部分&#xff1f;这时候不管你的CPU多么高级&#xff0c;显卡多么有信仰&#xff0c;正常的人类应该都会选择留下硬盘吧&#xff0c;因为硬盘里不仅住着…...