模拟实现消息队列项目(系列4) -- 服务器模块(内存管理)

目录

前言

1. 创建MemoryDataCenter

2. 封装Exchange 和 Queue方法

3. 封装Binding操作

4. 封装Message操作

4.1 封装消息中心集合messageMap

4.2 封装消息与队列的关系集合queueMessageMap的操作

5. 封装未确认消息集合waitMessage的操作

6. 从硬盘中恢复数据到内存中

7. MemoryDataCenter单元测试

结语


前言

        上一节我们总结了服务器模块的硬盘管理,将交换机,队列,绑定存书到Sqlite数据库中,将消息按照队进行创建文件存储在本地硬盘中.并且封装了对于数据库和文件的各种操作.实现了持久化的效果,但是实际的消息存储/转发,主要靠内存的结构.对于消息队列来说,内存部分是更关键的,内存速度更快,可以达到更高的并发.本节就对内存管理进行封装.本项目全部代码已上传Gitee,链接放在文章末尾,欢迎大家访问!


1. 创建MemoryDataCenter

路径:mqserver.datacenter.MemoryDataCenter

考虑到多线程的原因,我们将HashMap替换成ConcurrentHashMap (对每个哈希桶进行加锁,相对来说是线程安全的)

@Data
public class MemoryDataCenter {// 1. 交换机  多线程环境下使用,使用ConcurrentHashMap会相对线程安全//         key:ExchangeName,value:Exchange对象private ConcurrentHashMap<String, Exchange> exchangeMap = new ConcurrentHashMap<>();// 2. 队列  key:QueueName,value:MSQueue对象private ConcurrentHashMap<String, MSQueue> queueMap = new ConcurrentHashMap<>();// 3. 绑定  key:ExchangeName,value:HashMap(key:QueueName,value:MSQueue对象)private ConcurrentHashMap<String,ConcurrentHashMap<String, Binding>> bindingsMap = new ConcurrentHashMap<>();// 4. 消息  key:MessageID,value:Message对象private ConcurrentHashMap<String, Message> messageMap = new ConcurrentHashMap<>();// 5. 消息和队列的映射关系 HashMap: key:QueueName,value:LinkedList(Message对象)private ConcurrentHashMap<String, LinkedList<Message>> queueMessageMap = new ConcurrentHashMap<>();// 6. 未确认的消息  HashMap: key:QueueName,value:HashMap(key:MessageID,value:Message对象)private ConcurrentHashMap<String,ConcurrentHashMap<String, Message>> queueMessageWaitAckMap = new ConcurrentHashMap<>();
}

2. 封装Exchange 和 Queue方法

主要就是插入和获取数据以及删除

   /*** 1. 针对内存中的交换机,队列设置操作*/public void insertExchange(Exchange exchange) {exchangeMap.put(exchange.getName(), exchange);System.out.println("[MemoryDataCenter] 新交换机添加成功! exchangeName=" + exchange.getName());}public Exchange getExchange(String exchangeName) {return exchangeMap.get(exchangeName);}public void deleteExchange(String exchangeName) {exchangeMap.remove(exchangeName);System.out.println("[MemoryDataCenter] 交换机删除成功! exchangeName=" + exchangeName);}public void insertQueue(MSQueue queue) {queueMap.put(queue.getName(), queue);System.out.println("[MemoryDataCenter] 新队列添加成功! queueName=" + queue.getName());}public MSQueue getQueue(String queueName) {return queueMap.get(queueName);}public void deleteQueue(String queueName) {queueMap.remove(queueName);System.out.println("[MemoryDataCenter] 队列删除成功! queueName=" + queueName);}

3. 封装Binding操作

这里呢之所以将绑定的操作单独列举出来,是因为存储绑定信息的数据结构是相对比较复杂的,是嵌套的HashMap.

对于插入绑定信息:

1, 首先按照交换机的名字进行查找,如果查找不到就进行创建一个HashMap的数据结构存储到含有绑定信息的HashMap中,如果存在的话在按照队列名字进行查找绑定信息,如果查找到了,说明改绑定信息已经插入过就不要进行插入了,如果没找到就进行插入操作.

2. 在上述查找和插入的操作比并不是原子的,所以我们要给是上述操作,按照bindingMap进行加锁.以保证我们的线程操作是安全的.

下述是相关对于绑定的操作的代码:

    /*** 2. 针对绑定进行操作*//*** 2.1插入绑定信息* @param binding* @throws MqException*/public void insertBinding(Binding binding) throws MqException {
//        ConcurrentHashMap<String, Binding> bindingMap = bindingsMap.get(binding.getExchangeName());
//        if (bindingMap == null) {
//            bindingMap = new ConcurrentHashMap<>();
//            bindingsMap.put(binding.getExchangeName(), bindingMap);
//        }// 先使用 exchangeName 查一下, 对应的哈希表是否存在. 不存在就创建一个.ConcurrentHashMap<String, Binding> bindingMap = bindingsMap.computeIfAbsent(binding.getExchangeName(),k -> new ConcurrentHashMap<>());synchronized (bindingMap) {// 再根据 queueName 查一下目前的绑定的交换机绑定的是否是当前传入的队列. 如果已经存在(存在相同的绑定关系了,就不需要进行传入), 就抛出异常. 不存在才能插入.if (bindingMap.get(binding.getQueueName()) != null) {throw new MqException("[MemoryDataCenter] 绑定已经存在! exchangeName=" + binding.getExchangeName() +", queueName=" + binding.getQueueName());}// 最后将绑定关系传入到bingMap中bindingMap.put(binding.getQueueName(), binding);}System.out.println("[MemoryDataCenter] 新绑定添加成功! exchangeName=" + binding.getExchangeName()+ ", queueName=" + binding.getQueueName());}/*** 2.2 获取绑定1: 根据exchangeName, queueName 获取唯一的绑定* @param exchangeName* @param queueName*/public Binding getBinding(String exchangeName, String queueName){ConcurrentHashMap<String, Binding> bindingMap = bindingsMap.get(exchangeName);if (bindingMap == null){return null;}synchronized (bindingMap){// 防止当别的操作删除了这个队列的绑定信息,而导致的线程错误return bindingMap.get(queueName);}}/*** 2.3 获取绑定2: 根据exchangeName 查询所有绑定* @param exchangeName* @return*/public ConcurrentHashMap<String, Binding> getBindings(String exchangeName) throws MqException {if (bindingsMap.get(exchangeName) == null){return null;}return bindingsMap.get(exchangeName);}/*** 2.4 删除绑定关系(单个) 一个交换机对应的单个队列的绑定关系* @param binding* @throws MqException*/public void deleteBinding(Binding binding) throws MqException {ConcurrentHashMap<String, Binding> bindingMap = bindingsMap.get(binding.getExchangeName());if (bindingMap == null) {// 该交换机没有绑定任何队列. 报错.throw new MqException("[MemoryDataCenter] 绑定不存在! exchangeName=" + binding.getExchangeName()+ ", queueName=" + binding.getQueueName());}bindingMap.remove(binding.getQueueName());System.out.println("[MemoryDataCenter] 绑定删除成功! exchangeName=" + binding.getExchangeName()+ ", queueName=" + binding.getQueueName());}/*** 2.5 删除绑定关系(多个) 1个交换机对应的多个队列的绑定关系.*/public void deleteBinding(String exchangeName){bindingsMap.remove(exchangeName);}

4. 封装Message操作

4.1 封装消息中心集合messageMap

  • 1. 添加消息到消息中心
  • 2. 根据消息ID查询消息
  • 3. 根据消息ID删除消息
 /*** 3. 针对消息进行操作*//*** 3.1 添加消息* @param message*/public void addMessage(Message message) {messageMap.put(message.getMessageID(), message);System.out.println("[MemoryDataCenter] 新消息添加成功! messageId=" + message.getMessageID());}/*** 3.2 根据 id 查询消息* @param messageId* @return*/public Message getMessage(String messageId) {return messageMap.get(messageId);}/*** 3.3 根据 id 删除消息* @param messageId*/public void removeMessage(String messageId) {messageMap.remove(messageId);System.out.println("[MemoryDataCenter] 消息被移除! messageId=" + messageId);}

4.2 封装消息与队列的关系集合queueMessageMap的操作

  • 1. 发送消息到指定队列名字的队列
  • 2. 从指定队列中获取消息集合
  • 3. 获取指定队列名字队列中消息的个数
 /*** 4 针对消息和队列的关系进行操作*//*** 4.1 发送消息到指定队列* @param queue* @param message*/public void sendMessage(MSQueue queue, Message message) {// 先根据队列的名字, 找到该队列对应的消息链表.// 先根据队列的名字进行查询,查不到就进行创建该队列对应的链表  // computeIfAbsent线程安全的LinkedList<Message> messages = queueMessageMap.computeIfAbsent(queue.getName(),k-> new LinkedList<>());// 再把数据加到 messages 里面synchronized (messages) {// 对该队列进行添加的时候需要进行加锁messages.add(message);}// 在这里把该消息也往消息中心中插入一下. 假设如果 message 已经在消息中心存在, 重复插入也没关系.// 主要就是相同 messageId, 对应的 message 的内容一定是一样的. (服务器代码不会对 Message 内容做修改 basicProperties 和 body)addMessage(message);System.out.println("[MemoryDataCenter] 消息被添加到队列中! messageId=" + message.getMessageID());}/*** 4.2 从指定队列名字中进行提取信息* @param queueName* @return*/public Message pollMessage(String queueName){LinkedList<Message> messages = queueMessageMap.get(queueName);// 队列中没有信息if (messages == null){System.out.println("[MemoryDataCenter] 该队列中没有信息! queueName=" + queueName);return null;}// 将队列进行头删除(提取信息)synchronized (messages){if (messages.size() == 0){System.out.println("[MemoryDataCenter] 该队列中没有信息! queueName=" + queueName);return null;}Message currentMessage = messages.remove(0); System.out.println("[MemoryDataCenter] 消息已经从队列中取出! queueName=" + queueName + ", MessageID=" + currentMessage.getMessageID() );return currentMessage;}}/*** 4.3 获取指定队列名字中消息的个数* @param queueName* @return*/public int getMessageCount(String queueName){LinkedList<Message> messages = queueMessageMap.get(queueName);// 队列中没有信息if (messages == null){System.out.println("[MemoryDataCenter] 该队列中没有信息! queueName=" + queueName);return 0;}// 将队列进行头删除(提取信息)synchronized (messages){if (messages.size() == 0){System.out.println("[MemoryDataCenter] 该队列中没有信息! queueName=" + queueName);return 0;}return messages.size();}}

5. 封装未确认消息集合waitMessage的操作

  • 1. 添加消息到等待确认队列
  • 2. 从指定未确认队列中删除消息
  • 3. 根据指定的消息ID与未确认队列名字获取消息内容
/*** 5. 未确认消息Map的操作*//*** 5.1 添加消息到指定等待确认队列* @param queueName* @param message*/public void addMessageWaitAck(String queueName, Message message){ConcurrentHashMap<String,Message> waitMessage = queueMessageWaitAckMap.computeIfAbsent(queueName, k-> new ConcurrentHashMap<>());waitMessage.put(message.getMessageID(),message);System.out.println("[MemoryDataCenter] 消息进入等待确认队列! messageID=" + message.getMessageID());}/*** 5.2 从指定的未确认消息队列中进行删除消息* @param queueName* @param messageId*/public void removeMessageWaitAck(String queueName, String messageId){ConcurrentHashMap<String,Message> waitMessage = queueMessageWaitAckMap.get(queueName);if (waitMessage == null){System.out.println("[MemoryDataCenter] 该队列为空! queueName=" + queueName);return;}waitMessage.remove(messageId);System.out.println("[MemoryDataCenter] 消息已经从等待确认队列中移除! messageId=" + messageId);}/*** 5.3 根据指定消息ID从队列中进行获取信息* @param queueName* @param messageId* @return*/public Message geMessageWaitAck(String queueName, String messageId){ConcurrentHashMap<String,Message> waitMessage = queueMessageWaitAckMap.get(queueName);if (waitMessage == null){System.out.println("[MemoryDataCenter] 该队列为空! queueName=" + queueName);return null;}return waitMessage.get(messageId);}

6. 从硬盘中恢复数据到内存中

使用之前封装过的diskDataCenter进行恢复数据.

1. 清空当前内存数据结构中的数据

2. 恢复所有的交换机,队列,绑定,消息数据,恢复消息数据的时候,要将消息中心和消息与队列的映射进行恢复.

/*** 6. 从硬盘中恢复数据到内存中 (使用之前封装好的管理硬盘的类进行实现)*/public void recovery(DiskDataCenter diskDataCenter) throws IOException, MqException, ClassNotFoundException {// 1. 清空内存中各种数据信息queueMap.clear();exchangeMap.clear();bindingsMap.clear();messageMap.clear();queueMessageMap.clear();// 2. 恢复所有的交换机信息List<Exchange> exchanges = diskDataCenter.selectAllExchange();for (Exchange exchange :exchanges) {exchangeMap.put(exchange.getName(),exchange);}// 3. 恢复所有的队列信息List<MSQueue> queues = diskDataCenter.selectAllMSQueue();for (MSQueue msQueue :queues) {queueMap.put(msQueue.getName(),msQueue);}// 4. 恢复所有的绑定数据List<Binding> bindings = diskDataCenter.selectAllBinding();for (Binding binding: bindings){ConcurrentHashMap<String,Binding> bindingMap = bindingsMap.computeIfAbsent(binding.getExchangeName(), k-> new ConcurrentHashMap<>());bindingMap.put(binding.getQueueName(),binding);}// 4. 恢复所有的消息数据// 4.1 遍历所有的队列// List<MSQueue> queues = diskDataCenter.selectAllMSQueue();for (MSQueue msQueue:queues) {LinkedList<Message> messages = diskDataCenter.loadAllMessageFromQueue(msQueue.getName());// 4.2 将获取的消息进行进行加入到队列queueMessageMap.put(msQueue.getName(),messages);// 4.3 将消息添加上到消息中心for (Message message : messages) {messageMap.put(message.getMessageID(),message);}}

7. MemoryDataCenter单元测试

package com.example.demo.mqserver.datacenter;import com.example.demo.DemoApplication;
import com.example.demo.common.MqException;
import com.example.demo.mqserver.core.*;
import org.apache.tomcat.util.http.fileupload.FileUtils;
import org.junit.jupiter.api.AfterEach;
import org.junit.jupiter.api.Assertions;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.test.context.SpringBootTest;import java.io.File;
import java.io.IOException;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.ConcurrentHashMap;import static org.junit.jupiter.api.Assertions.*;/*** Created with IntelliJ IDEA.* Description:* User: YAO* Date: 2023-07-31* Time: 10:30*/
@SpringBootTest
class MemoryDataCenterTest {MemoryDataCenter memoryDataCenter = null;@BeforeEachvoid setUp() {memoryDataCenter = new MemoryDataCenter();}@AfterEachvoid tearDown() {memoryDataCenter = null;}// 创建一个测试交换机private Exchange createTestExchange(String exchangeName) {Exchange exchange = new Exchange();exchange.setName(exchangeName);exchange.setType(ExchangeType.DIRECT);exchange.setAutoDelete(false);exchange.setDurable(true);return exchange;}// 创建一个测试队列private MSQueue createTestQueue(String queueName) {MSQueue queue = new MSQueue();queue.setName(queueName);queue.setDurable(true);queue.setExclusive(false);queue.setAutoDelete(false);return queue;}/*** 1. 针对交换机进行操作*/@Testpublic void testExchange(){// 1. 创建交换机进行插入Exchange expectExchange = createTestExchange("testExchange");memoryDataCenter.insertExchange(expectExchange);// 2. 查询交换机Exchange actualExchange = memoryDataCenter.getExchange("testExchange");// 比较内存中的引用是否是同一个引用Assertions.assertEquals(expectExchange,actualExchange);// 3. 删除交换机memoryDataCenter.deleteExchange("testExchange");// 4. 查询交换机,比较结果actualExchange = memoryDataCenter.getExchange("testExchange");Assertions.assertNull(actualExchange);}/*** 2. 针对队列进行操作*/@Testpublic void testQueue(){// 1. 创建交换机进行插入MSQueue expectQueue = createTestQueue("testQueue");memoryDataCenter.insertQueue(expectQueue);// 2. 查询交换机MSQueue actualQueue = memoryDataCenter.getQueue("testQueue");// 比较内存中的引用是否是同一个引用Assertions.assertEquals(expectQueue,actualQueue);// 3. 删除交换机memoryDataCenter.deleteQueue("testQueue");// 4. 查询交换机,比较结果actualQueue = memoryDataCenter.getQueue("testQueue");Assertions.assertNull(actualQueue);}/*** 3. 针对绑定进行测试*/@Testpublic void testBinding() throws MqException {// 1.创建绑定并加入到集合中Binding expectedBinding = new Binding();expectedBinding.setExchangeName("testExchange");expectedBinding.setQueueName("testQueue");memoryDataCenter.insertBinding(expectedBinding);// 2. 查询绑定(单个)Binding actualBinding = memoryDataCenter.getBinding("testExchange","testQueue");Assertions.assertEquals(expectedBinding,actualBinding);// 2.1 查询所有的绑定ConcurrentHashMap<String, Binding> bindingMap = memoryDataCenter.getBindings("testExchange");Assertions.assertEquals(1, bindingMap.size());Assertions.assertEquals(expectedBinding, bindingMap.get("testQueue"));// 3. 删除绑定memoryDataCenter.deleteBinding("testExchange");actualBinding = memoryDataCenter.getBinding("testExchange","testQueue");Assertions.assertNull(actualBinding);bindingMap = memoryDataCenter.getBindings("testExchange");Assertions.assertNull(bindingMap);}private Message createTestMessage(String content) {Message message = Message.createMessageWithId("testRoutingKey", null, content.getBytes());return message;}/*** 4. 针对消息进行测试*/@Testpublic void testMessage(){// 1. 创建消息并插入Message expectedMessage = createTestMessage("testMessage");memoryDataCenter.addMessage(expectedMessage);// 2. 查询消息并比较Message actualMessage = memoryDataCenter.getMessage(expectedMessage.getMessageID());Assertions.assertEquals(expectedMessage, actualMessage);// 4. 删除消息memoryDataCenter.removeMessage(expectedMessage.getMessageID());// 5. 查询消息并比较actualMessage = memoryDataCenter.getMessage(expectedMessage.getMessageID());Assertions.assertNull(actualMessage);}/*** 5. 测试将消息发送到对列中*/@Testpublic void sendMessage(){// 1. 创建一个队列. 创建10条消息,进行插入到队列MSQueue expectQueue = createTestQueue("testQueue");List<Message> expectMessage = new ArrayList<>();for (int i = 0; i < 10; i++) {Message message = createTestMessage("testMessage" + i);memoryDataCenter.sendMessage(expectQueue,message);expectMessage.add(message);}// 2.从队列进行取出消息List<Message> actualMessage = new ArrayList<>();while (true){Message message = memoryDataCenter.pollMessage("testQueue");if (message == null){break;}actualMessage.add(message);}// 3. 比较消息前后是否一致Assertions.assertEquals(expectMessage.size(),actualMessage.size());for (int i = 0; i < expectMessage.size(); i++) {Assertions.assertEquals(expectMessage.get(i),actualMessage.get(i));}}/*** 6. 测试未被确认的消息*/@Testpublic void testMessageWaitAck(){// 1. 创建消息,插入到未被确认的队列中Message expectedMessage = createTestMessage("expectedMessage");memoryDataCenter.addMessageWaitAck("testQueue", expectedMessage);// 2. 获取消息从未被确认的队列中Message actualMessage = memoryDataCenter.geMessageWaitAck("testQueue", expectedMessage.getMessageID());Assertions.assertEquals(expectedMessage, actualMessage);// 3. 从未被确认的队列中进行删除消息memoryDataCenter.removeMessageWaitAck("testQueue", expectedMessage.getMessageID());// 4. 比较删除之后的队列是否还有消息actualMessage = memoryDataCenter.geMessageWaitAck("testQueue", expectedMessage.getMessageID());Assertions.assertNull(actualMessage);}/*** 7. 测试从硬盘中恢复数据到内存*/@Testpublic void testRecovery() throws IOException, MqException, ClassNotFoundException {// 由于后续需要进行数据库操作, 依赖 MyBatis. 就需要先启动 SpringApplication, 这样才能进行后续的数据库操作.DemoApplication.context = SpringApplication.run(DemoApplication.class);// 1. 在硬盘上构造好数据DiskDataCenter diskDataCenter = new DiskDataCenter();diskDataCenter.init();// 构造交换机Exchange expectedExchange = createTestExchange("testExchange");diskDataCenter.insertExchange(expectedExchange);// 构造队列MSQueue expectedQueue = createTestQueue("testQueue");diskDataCenter.insertQueue(expectedQueue);// 构造绑定Binding expectedBinding = new Binding();expectedBinding.setExchangeName("testExchange");expectedBinding.setQueueName("testQueue");expectedBinding.setBindingKey("testBindingKey");diskDataCenter.insertBinding(expectedBinding);// 构造消息Message expectedMessage = createTestMessage("testContent");diskDataCenter.sendMessage(expectedQueue, expectedMessage);// 2. 执行恢复操作memoryDataCenter.recovery(diskDataCenter);// 3. 对比结果Exchange actualExchange = memoryDataCenter.getExchange("testExchange");Assertions.assertEquals(expectedExchange.getName(), actualExchange.getName());Assertions.assertEquals(expectedExchange.getType(), actualExchange.getType());Assertions.assertEquals(expectedExchange.isDurable(), actualExchange.isDurable());Assertions.assertEquals(expectedExchange.isAutoDelete(), actualExchange.isAutoDelete());MSQueue actualQueue = memoryDataCenter.getQueue("testQueue");Assertions.assertEquals(expectedQueue.getName(), actualQueue.getName());Assertions.assertEquals(expectedQueue.isDurable(), actualQueue.isDurable());Assertions.assertEquals(expectedQueue.isAutoDelete(), actualQueue.isAutoDelete());Assertions.assertEquals(expectedQueue.isExclusive(), actualQueue.isExclusive());Binding actualBinding = memoryDataCenter.getBinding("testExchange", "testQueue");Assertions.assertEquals(expectedBinding.getExchangeName(), actualBinding.getExchangeName());Assertions.assertEquals(expectedBinding.getQueueName(), actualBinding.getQueueName());Assertions.assertEquals(expectedBinding.getBindingKey(), actualBinding.getBindingKey());Message actualMessage = memoryDataCenter.pollMessage("testQueue");Assertions.assertEquals(expectedMessage.getMessageID(), actualMessage.getMessageID());Assertions.assertEquals(expectedMessage.getRoutingKey(), actualMessage.getRoutingKey());Assertions.assertEquals(expectedMessage.getDeliverMode(), actualMessage.getDeliverMode());Assertions.assertArrayEquals(expectedMessage.getBody(), actualMessage.getBody());// 4. 清理硬盘的数据, 把整个 data 目录里的内容都删掉(包含了 meta.db 和 队列的目录).DemoApplication.context.close();File dataDir = new File("./data");FileUtils.deleteDirectory(dataDir);}
}


结语

        以上内容就是针对内存管理的封装,主要是设计了6中数据机构进行存储交换机 队列 绑定 消息 消息和队列的映射 未确认信息.后续对数据进行操作的时候会更加具有效率.这样我们虚拟主机中两大核心部分:硬盘管理和内存管理都总结完成,下一节会对上述两种操作进一步封装到(VirtualHost)中,然后正式的提出消息队列服务器BrokerServer这个概念,对其进行完善和功能封装.请持续关注,谢谢!!!

完整的项目代码已上传Gitee,欢迎大家访问.👇👇👇

模拟实现消息队列https://gitee.com/yao-fa/advanced-java-ee/tree/master/My-mq

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

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

相关文章

Leetcode 每日一题 - 删除有序数组中的重复项题 #算法 #Java

1.1 题目 给你一个 升序排列 的数组 nums &#xff0c;请你 原地 删除重复出现的元素&#xff0c;使每个元素 只出现一次 &#xff0c;返回删除后数组的新长度。元素的 相对顺序 应该保持 一致 。然后返回 nums 中唯一元素的个数。 考虑 nums 的唯一元素的数量为 k &#xff…

行业追踪,2023-08-07

自动复盘 2023-08-07 凡所有相&#xff0c;皆是虚妄。若见诸相非相&#xff0c;即见如来。 k 线图是最好的老师&#xff0c;每天持续发布板块的rps排名&#xff0c;追踪板块&#xff0c;板块来开仓&#xff0c;板块去清仓&#xff0c;丢弃自以为是的想法&#xff0c;板块去留让…

虹科分享 | 新时代“救命神器”:看AR眼镜如何应用于紧急救险场景

从工业时代到如今迎来的“体验时代”&#xff0c;体验即内容&#xff0c;5G、AI、空间计算技术的突破&#xff0c;为各行各业创建了丰富的内容体验模式&#xff0c;让人们能够听之、触之、与之交互。AR是体验时代最具潜力的新技术&#xff0c;在“应急”场景中更是成为了我们在…

JSP实训项目设计报告—MVC简易购物商城

JSP实训项目设计报告—MVC简易购物商城 文章目录 JSP实训项目设计报告—MVC简易购物商城设计目的设计要求设计思路系统要求单点登录模块商品展示模块购物车展示模块 概要设计Model层View层Controller层 详细设计Model层View层登录界面系统主界面 Controller层 系统运行效果项目…

云计算——ACA学习 云计算概述

作者简介&#xff1a;一名云计算网络运维人员、每天分享网络与运维的技术与干货。 座右铭&#xff1a;低头赶路&#xff0c;敬事如仪 个人主页&#xff1a;网络豆的主页​​​​​ 目录 写在前面 上章回顾 本章简介 本章目标 一.云计算产生背景 1.信息时代的重点变革…

【VUE】前端实现防篡改的水印

效果 水印的作用 图片加水印的操作一般是由后端来完成&#xff0c;有些站点保护的知识产权的类型可能比较多&#xff0c;不仅仅是图片&#xff0c;可能还有视频、文字等等&#xff0c;对于不同类型的对象添加水印后端操作比较复杂&#xff0c;所有有些站点逐步的让前端去进行水…

认识所有权

专栏简介&#xff1a;本专栏作为Rust语言的入门级的文章&#xff0c;目的是为了分享关于Rust语言的编程技巧和知识。对于Rust语言&#xff0c;虽然历史没有C、和python历史悠远&#xff0c;但是它的优点可以说是非常的多&#xff0c;既继承了C运行速度&#xff0c;还拥有了Java…

前沿分享-鱼形机器人

可能并不太前沿了&#xff0c;是21年底的新闻了&#xff0c;但是看见了就顺便发一下吧。 大概就是&#xff0c;通过在pH响应型水凝胶中编码不同的膨胀速率而构建了一种环境适应型变形微机器人,让微型机器人直接向癌细胞输送药物从而减轻药物带来副作用。 技术原理是&#xff0c…

拦截器对接口细粒度权限校验

文章目录 一、逻辑分析二、校验规则1.规则类型2.规则划分3.规则配置信息4.规则案例说明5.规则加载 三、拦截器定义1.自定义拦截器2.注册拦截器 四、获取请求参数1.获取get提交方式参数2.获取post提交方式参数&#xff08;1&#xff09;定义RequestWrapper类&#xff08;2&#…

Flink正常消费一段时间后,大量反压,看着像卡住了,但又没有报错。

文章目录 前言一、原因分析二、解决方案 前言 前面我也有提到&#xff0c;发现flink运行一段时间后&#xff0c;不再继续消费的问题。这个问题困扰了我非常久&#xff0c;一开始也很迷茫。又因为比较忙&#xff0c;所以一直没有时间能够去寻找答案&#xff0c;只是通过每天重启…

IDEA中maven项目失效,pom.xml文件橙色/橘色

IDEA中maven项目失效&#xff0c;pom.xml文件橙色/橘色 IDEA中Maven项目失效 IDEA中创建的maven项目中的文件夹都变成普通格式&#xff0c;pom.xml变成橙色 右键点击橙色的pom.xml文件&#xff0c;选择add as maven project maven项目开始重新导入相应依赖&#xff0c;恢复…

字符串查找匹配算法

概述 字符串匹配&#xff08;查找&#xff09;是字符串的一种基本操作&#xff1a;给定带匹配查询的文本串S和目标子串T&#xff0c;T也叫做模式串。在文本S中找到一个和模式T相符的子字符串&#xff0c;并返回该子字符串在文本中的位置。 暴力匹配 Brute Force Algorithm&a…

安全狗V3.512048版本绕过

安全狗安装 安全狗详细安装、遇见无此服务器解决、在windows中命令提示符中进入查看指定文件夹手动启动Apache_安全狗只支持 glibc_2.14 但是服务器是2.17_黑色地带(崛起)的博客-CSDN博客 安全狗 safedogwzApacheV3.5.exe 右键电脑右下角安全狗图标-->选择插件-->安装…

vscode中无法使用git解决方案

1 首先查看git安装目录 where git 2 找到bash.exe 的路径 比如&#xff1a;C:/Users/Wangzd/AppData/Local/Programs/Git/bin/bash 3 找到vscode的配置项setting.json 4 添加 "terminal.integrated.shell.windowns": "C:/Users/Wangzd/AppData/Local/Pr…

架构训练营学习笔记:6-2 微服务基础选型

基础选型 微服务基础设施架构 优先级 其中&#xff0c;核心 就是服务注册、服务发现、服务路由。 模式1-嵌入SDK 模式2-反向代理式 模式3-网络代理式&#xff08;Service Mesh&#xff09; 模式对比 常见微服务框架选择 嵌入SDK-dubbo Spring Cloud 反向代理式 APISIX …

跨境B2B2C多用户购物网站源码快速部署

​ 搭建跨境B2B2C多用户购物网站需要以下步骤&#xff1a; 1. 确定业务模式和定位&#xff1a;确定网站的业务模式&#xff0c;包括跨境B2B2C的商业模式以及目标用户定位。 2. 营业执照和域名注册&#xff1a;根据当地法律要求&#xff0c;注册一家具有法人资格的公司&#xff…

基于Citespace、vosviewer、R语言的文献计量学可视化分析技术及全流程文献可视化SCI论文高效写作方法

跨尺度预测模式&#xff08;The Model for Prediction Across Scales - MPAS&#xff09;是由洛斯阿拉莫斯实验室和美国国家大气研究中心(NCAR)共同开发&#xff0c;其由3个部分组成&#xff0c;分别称为 MPAS-A&#xff08;大气模型&#xff09;、MPAS-O&#xff08;海洋模型&…

一百四十一、Kettle——kettle8.2在Windows本地开启carte服务以及配置子服务器

一、目的 在kettle建好共享资源库后&#xff0c;为了给在服务器上部署kettle的carte服务躺雷&#xff0c;先在Windows本地测试一下怎么玩carte服务 二、Kettle版本以及在Windows本地安装路径 kettle版本是8.2 pdi-ce-8.2.0.0-342 kettle本地安装路径是D:\j…

全面讲解最小二乘法

常见的最小二乘法我们就不多说了&#xff0c;下面主要介绍一下最小二乘法的一些先进方法。 正则化的最小二乘法 在使用常见的最小二乘法进行回归分析时&#xff0c;常常会遇到过拟合的问题&#xff0c;也就是在训练数据集上表现的很好&#xff0c;但是在测试数据集上表现的很…

【Maven】常用命令、插件管理、私服nexus

【Maven】常用命令、插件管理、私服nexus 常用命令 插件管理 私服nexus Nexus3 配置私服 项目pom中的配置 发布时区分正式版、快照版 常用命令 Maven提供了一系列常用命令&#xff0c;用于构建、测试和管理项目。以下是一些常用的Maven命令示例&#xff1a; mvn clean:…