模拟实现消息队列项目(系列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
相关文章:
![](https://img-blog.csdnimg.cn/2a3c5e7cf5b046b683f1adf03894b108.png)
模拟实现消息队列项目(系列4) -- 服务器模块(内存管理)
目录 前言 1. 创建MemoryDataCenter 2. 封装Exchange 和 Queue方法 3. 封装Binding操作 4. 封装Message操作 4.1 封装消息中心集合messageMap 4.2 封装消息与队列的关系集合queueMessageMap的操作 5. 封装未确认消息集合waitMessage的操作 6. 从硬盘中恢复数据到内存中 7. Memo…...
![](https://img-blog.csdnimg.cn/1069e5d70fdd4c6f8780d363d8a7a3a1.png)
STM32 LoRa源码解读
目录结构: SX1278 |-- include | |-- fifo.h | |-- lora.h | |-- platform.h | |-- radio.h | |-- spi.h | |-- sx1276.h | |-- sx1276Fsk.h | |-- sx1276FskMisc.h | |-- sx1276Hal.h | |-- sx1276LoRa.h | -- sx1276LoRaMisc.h – src |-- fifo.c |-- lora.c |-- …...
![](https://www.ngui.cc/images/no-images.jpg)
【BASH】回顾与知识点梳理(十)
【BASH】回顾与知识点梳理 十 十. 文件的格式化与相关处理10.1 格式化打印: printf10.2 awk:好用的数据处理工具awk 的逻辑运算字符 10.3 文件比对工具diffcmppatch 10.4 文件打印准备: pr 该系列目录 --> 【BASH】回顾与知识点梳理&#…...
![](https://img-blog.csdnimg.cn/dc37326249bc488aabd74b9b45f7b57e.png)
【网络】应用层——HTTPS协议
🐱作者:一只大喵咪1201 🐱专栏:《网络》 🔥格言:你只管努力,剩下的交给时间! HTTPS协议 🍉HTTP的不安全性🍉认识HTTPS协议🍓加密解密ἵ…...
![](https://img-blog.csdnimg.cn/1f47e03579a64ccf9065d12d4e56c932.png)
Windows新版文件资源管理器经常在后台弹出的临时解决方案
禁用组策略自动刷新 运行gpedit.msc找到计算机配置->管理模板->系统->组策略找到 “关闭组策略的后台刷新”启用 参考 https://answers.microsoft.com/en-us/windows/forum/all/windows-11-most-recently-opened-explorer-window/26e097bd-1eba-4462-99bd-61597b5…...
![](https://img-blog.csdnimg.cn/7765de14db6e4064b76d6c4ab0d1e7e7.png#pic_center)
软考高项(八)项目整合管理 ★重点集萃★
👑 个人主页 👑 :😜😜😜Fish_Vast😜😜😜 🐝 个人格言 🐝 :🧐🧐🧐说到做到,言出必行&am…...
![](https://www.ngui.cc/images/no-images.jpg)
基于python+django开发的学生信息管理系统
基于pythondjangovue.js开发的学生信息管理系统,师弟的课程作业 功能介绍 平台采用B/S结构,后端采用主流的Python语言进行开发,前端采用主流的Vue.js进行开发。 功能包括:学生管理、班级管理、用户管理、日志管理、系统信息模块…...
![](https://img-blog.csdnimg.cn/a230d4ad2c65465e87f70e45fff3f566.png)
mysql的高级查询语句
目录 一、本文前言 二、高效查询方式 1)指定指字段进行查看 2)对字段进行去重查看 3)where条件查询 4)and 和 or 进行逻辑关系的增加 5)查询取值列表中的数据 6)between的引用 7)like…...
![](https://img-blog.csdnimg.cn/17d6bfea5dd04974bebcdd88be79903e.png)
04-8_Qt 5.9 C++开发指南_QTableWidget的使用
文章目录 1. QTableWidget概述2. 源码2.1 可视化UI设计2.2 程序框架2.3 qwintspindelegate.h2.4 qwintspindelegate.cpp2.5 mainwindow.h2.6 mainwindow.cpp 1. QTableWidget概述 QTableWidget是Qt中的表格组件类。在窗体上放置一个QTableWidget 组件后,可以在 PropertyEditor…...
![](https://img-blog.csdnimg.cn/4a3d2dc909d846f3ade748eb38789629.png#pic_center)
《golang设计模式》第二部分·结构型模式-01-适配器模式(Adapter)
文章目录 1. 概念1.1 角色1.2 应用场景1.2 类图 2. 代码示例2.1 设计2.2 代码2.3 示例类图 1. 概念 定义一个适配器,帮助原本不能实现接口的类“实现”该接口 1.1 角色 目标(Target):客户端调用的目标接口 被适配者(…...
![](https://img-blog.csdnimg.cn/img_convert/1b6627eedaba4dbe6b74dde00daa2cfa.png)
机器学习概述及其主要算法
目录 1、什么是机器学习 2、数据集 2.1、结构 3、算法分类 4、算法简介 4.1、K-近邻算法 4.2、贝叶斯分类 4.3、决策树和随机森林 4.4、逻辑回归 4.5、神经网络 4.6、线性回归 4.7、岭回归 4.8、K-means 5、机器学习开发流程 6、学习框架 1、什么是机器学习 机器…...
![](https://www.ngui.cc/images/no-images.jpg)
识jvm堆栈中一个数据类型是否为为引用类型,目前虚拟机实现中是如何做的?
调用栈里的引用类型数据是GC的根集合(root set)的重要组成部分;找出栈上的引用是GC的根枚举(root enumeration)中不可或缺的一环。 要看JVM选择用什么方式。通常这个选择会影响到GC的实现。 如果JVM选择不记录任何这种…...
![](https://www.ngui.cc/images/no-images.jpg)
Bug合集
这里会收藏后面所遇到的bug并附上具有参考的意义的博客,会持续更新 Java 1、SpringBoot升级2.6.0以上后,Swagger出现版本不兼容报错。 Failed to start bean ‘documentationPluginsBootstrapper‘; nested exception is java.lang.NullPo…...
![](https://img-blog.csdnimg.cn/3040ece234f0473294eee1ab5a54c1ca.png)
linux下.run安装脚本制作
1、安装文件(install.sh) PS: .run安装包内部执行脚本文件 2、资源文件(test.zip) PS: 待安装程序源文件 3、制作.run脚本(install.run) cat install.sh test.zip > install.run chmod ax install.run...
![](https://img-blog.csdnimg.cn/6527ace46f434303aacb5b8d71c2c949.png)
面试热题(翻转k个链表)
给你链表的头节点 head ,每 k 个节点一组进行翻转,请你返回修改后的链表。 k 是一个正整数,它的值小于或等于链表的长度。如果节点总数不是 k 的整数倍,那么请将最后剩余的节点保持原有顺序。 你不能只是单纯的改变节点内部的值&a…...
![](https://www.ngui.cc/images/no-images.jpg)
前端面试的性能优化部分(4)每天10个小知识点
目录 系列文章目录前端面试的性能优化部分(1)每天10个小知识点前端面试的性能优化部分(2)每天10个小知识点前端面试的性能优化部分(3)每天10个小知识点前端面试的性能优化部分(4)每天…...
![](https://www.ngui.cc/images/no-images.jpg)
el-checkbox修改选中和未选中的值
由于在checkbox选中的时候,默认的是为true和false。 后端定义的绑定字段不是为布尔值,而是为0,1 解决办法 <el-checkbox v-model"model.status" :true-label"1" :false-label"0"> </el-checkbox>…...
![](image/UDP-example.png)
完整版:TCP、UDP报文格式
目录 TCP报文格式 报文格式 报文示例 UDP报文格式 报文格式 报文示例 TCP报文格式 报文格式 图1 TCP首部格式 字段长度含义Source Port16比特源端口,标识哪个应用程序发送。Destination Port16比特目的端口,标识哪个应用程序接收。Sequence Numb…...
![](https://www.ngui.cc/images/no-images.jpg)
如何远程连接云服务器oracle数据库
要远程连接云服务器上的Oracle数据库,可以按照以下步骤进行操作: 1. 确保你的云服务器已经安装了Oracle数据库,并且启动了数据库服务。 2. 登录到云服务器的操作系统。可以使用SSH工具(如PuTTY)连接到云服务器,使用管理员权限登录…...
![](https://www.ngui.cc/images/no-images.jpg)
“深入剖析JVM内部机制:探秘Java虚拟机的运行原理“
标题:深入剖析JVM内部机制:探秘Java虚拟机的运行原理 摘要:本文将深入剖析Java虚拟机(JVM)的内部机制,探秘其运行原理。我们将从JVM的结构、内存管理、垃圾回收、即时编译等方面展开讨论,并通过…...
![](https://img-blog.csdnimg.cn/c5efe98204f94589a7086f8b25697d05.png)
尚品汇总结十:秒杀模块(面试专用)
1、需求分析 所谓“秒杀”,就是商家发布一些超低价格的商品,所有买家在同一时间网上抢购的一种销售方式。通俗一点讲就是商家为促销等目的组织的网上限时抢购活动。由于商品价格低廉,往往一上架就被抢购一空,有时只用一秒钟。 秒…...
![](https://img-blog.csdnimg.cn/10d46553089841bb88ca77ab55e4a6fa.png)
什么是设计模式?
目录 概述: 什么是模式!! 为什么学习模式!! 模式和框架的比较: 设计模式研究的历史 关于pattern的历史 Gang of Four(GoF) 关于”Design”Pattern” 重提:指导模式设计的三个概念 1.重用(reuse)…...
![](https://img-blog.csdnimg.cn/4bdd7ec23e7a4f35b97e12d7556257b9.png)
Node.js |(三)Node.js API:path模块及Node.js 模块化 | 尚硅谷2023版Node.js零基础视频教程
学习视频:尚硅谷2023版Node.js零基础视频教程,nodejs新手到高手 文章目录 📚path模块📚Node.js模块化🐇介绍🐇模块暴露数据⭐️模块初体验⭐️暴露数据 🐇导入文件模块🐇导入文件夹的…...
![](https://img-blog.csdnimg.cn/7bd1202286764a43bd53396456b3ff77.png)
Netty自定义编码解码器
上次通信的时候用的是自带的编解码器,今天自己实现一下自定义的。 1、自定义一下协议 //协议类 Data public class Protocol<T> implements Serializable {private Long id System.currentTimeMillis();private short msgType;// 假设1为请求 2为响应privat…...
![](https://img-blog.csdnimg.cn/cce213973d8a44b28dbdae6ff3708cd0.png)
HOperatorSet.OpenFramegrabber “GigEVision“
HOperatorSet.OpenFramegrabber "GigEVision"访问失败 直接跳出 但其他算子可以访问 重装halcon x86...
![](https://www.ngui.cc/images/no-images.jpg)
图的遍历DFSBFS-有向图无向图
西江月・证明 即得易见平凡,仿照上例显然。留作习题答案略,读者自证不难。 反之亦然同理,推论自然成立。略去过程Q.E.D.,由上可知证毕。 有向图的遍历可以使用深度优先搜索(DFS)和广度优先搜索(…...
![](https://img-blog.csdnimg.cn/990cfdfd1d0142f49b52d7e3a23c2bdb.png)
【NLP】深入浅出全面回顾注意力机制
深入浅出全面回顾注意力机制 1. 注意力机制概述2. 举个例子:使用PyTorch带注意力机制的Encoder-Decoder模型3. Transformer架构回顾3.1 Transformer的顶层设计3.2 Encoder与Decoder的输入3.3 高并发长记忆的实现self-attention的矩阵计算形式多头注意力(…...
![](https://www.ngui.cc/images/no-images.jpg)
Linux应用编程的read函数和Linux驱动编程的read函数的区别
Linux应用编程的read函数用于从文件描述符(文件、管道、套接字等)中读取数据。它的原型如下: ssize_t read(int fd, void *buf, size_t count);其中,fd参数是文件描述符,buf是用于存储读取数据的缓冲区,co…...
![](https://img-blog.csdnimg.cn/eefbf54241e04c9aa98e887e3493e288.png)
Kubernetes(K8s)从入门到精通系列之十:使用 kubeadm 创建一个高可用 etcd 集群
Kubernetes K8s从入门到精通系列之十:使用 kubeadm 创建一个高可用 etcd 集群 一、etcd高可用拓扑选项1.堆叠(Stacked)etcd 拓扑2.外部 etcd 拓扑 二、准备工作三、建立集群1.将 kubelet 配置为 etcd 的服务管理器。2.为 kubeadm 创建配置文件…...
![](https://img-blog.csdnimg.cn/img_convert/421de6bdee5da64a3d0268ab4ede2698.png)
使用动态规划实现错排问题-2023年全国青少年信息素养大赛Python复赛真题精选
[导读]:超平老师计划推出《全国青少年信息素养大赛Python编程真题解析》50讲,这是超平老师解读Python编程挑战赛真题系列的第15讲。 全国青少年信息素养大赛(原全国青少年电子信息智能创新大赛)是“世界机器人大会青少年机器人设…...
![](/WebClient/static//BC.P.plus/img/profile_small.jpg)
手机微信客户端网站建设/成都网站设计公司
ajax 毕竟是异步的 所以动态加载出来的数据 难免遇到 css 或者 js 失效的问题,所以要动态加载 css ji等文件了1.公共方法 load//动态加载 js /cssfunction loadjscssfile(filename, filetype) {if (filetype "js") { //判定文件类型var fileref documen…...
![](/images/no-images.jpg)
微信企业网站源码下载/不需要验证码的广告平台
//Retry机制public static class Retry{/// <summary>/// 重试零个参数无返回值的方法/// </summary>/// <param name"action">执行方法方法</param>/// <param name"retryInterval">重试间隔</param>/// <param n…...
![](/images/no-images.jpg)
论坛类型的网站怎么做/官网站内推广内容
一、今天在写东西时使用了mysl中的length()方法来检测数据长度,发现一个问题:当变量为纯英文字符时没有问题,但是中间夹杂着汉字时,这个结果就不一样了。这是因为在mysql中一个汉字的length为3导致的。 二…...
![](http://pbl6hpz34.bkt.clouddn.com/1540995293104nr1ktifl.png?imageslim)
https下安装wordpress/对网站进行seo优化
资源整理。这次太久没更新,最近比较忙。老规矩拆成两篇。 Coding: 1.比较独立,部分汇集和系统发育知识的生态位模型。“Niche estimation above and below the species level. Trends in Ecology and Evolution”论文的附件。 phyloENM TREE 2.用于生态预…...
![](https://images.cnblogs.com/OutliningIndicators/ExpandedBlockStart.gif)
一个网站需要怎么做/免费做网站网站
输入一个无向图<V,E> V<1e5, E<3e5 现在另外给k条边(u1,vs[k],wy[k]) 问在不影响从结点1出发到所有结点的最短路的前提下,最多可以删除k条边的多少条 跑最短路的时候维护或者统计就好了 一开始用spfa.然后TLE 45...好久没写 Dij堆优化 ... p.s.优…...
![](/images/no-images.jpg)
网站有几个后台/网站外包一般多少钱啊
更多编程教程请到:菜鸟教程 https://www.piaodoo.com/ 友情链接: 高州阳光论坛https://www.hnthzk.com/人人影视http://www.sfkyty.com/这两个函数主要提供,基于字典的访问局部和全局变量的方式。在理解这两个函数时,首先来理解…...