消息队列项目创建第二部分

news2025/1/19 23:07:16

消息队列项目创建第二部分

    • 一、在硬盘上存储信息
      • 使用文件存储消息
        • 具体存放策略
      • 垃圾回收(JVM)
      • 创建文件管理类——MessageFileManger
        • 创建统计文件数据和文件
        • 统计文件的读写操作
        • 创建消息对应的文件和目录
          • 创建一个统一处理异常
      • 消息文件的读写
        • 消息的序列化和反序列化
      • 发送信息和获取信息
      • 逻辑删除信息
      • 读取文件消息加载到内存中
      • 垃圾回收---》复制算法
          • `MessageFileManger`总结:
      • 测试功能

前面我们已经将三个类和三个类的操作模块进行的单元测试全部通过。只有消息类未进行测试,这是因为消息是特殊的,需要持久化存储。接下来创建第二个消息队列比较重要的部分(数据持久化):

  1. 实现消息存储在硬盘上
    • 对于消息的操作不需要大量的增删改查
    • 对于文件的操作效率要比数据库高

一、在硬盘上存储信息

使用文件存储消息

  1. 消息操作可能不会涉及倒复杂的增删改查。
  2. 消息的数量可能会非常多,而对于数据库的访问效率并不是非常高的,所以这里我们决定将消息存储在文件中。
具体存放策略
  • 在数据库所在的文件夹中创建一些文件或文件夹来存储对应的消息。

  • 在每个文件夹中分配两个文件用于存储不一样的数据

    ①保存消息的具体内容data.txt
    在这里插入图片描述

    • 二进制格式
    • 每个消息分为两部分
      • 前4字节表示Message的长度
      • 后面的就是Message内容

    在这里插入图片描述

    定位消息使用Message偏移量

    1. offsetBeg 开始
    2. offsetEnd 结束

    ②保存消息的统计信息state.txt

    • 使用文本文件存储,里面包含两列,使用’\t’分割
      • 第一列包含总消息数目
      • 第二列表示有效消息数目
例如:
2000\t1500

垃圾回收(JVM)

垃圾回收机制一共有五种方法:

  1. 引用计数
    • 跟随每个对象的引用次数,如果当前对象引用次数为0,表示不再引用该对象,可以被回收。
  2. 标记清除
    • 通过根节点遍历,标记所有可达对象,然后对其他直接进行清除操作。
  3. 复制算法
    • 将内存空间划分为两个大小相同的内存,每次只使用一半的内存,然后当内存使用超过一半时将不是垃圾的对象拷贝到另一半的空间中,然后清空旧区域所有对象。
  4. 标记整理
    • 结合了标记清除和复制算法的优点,首先标记可达对象,再将存活的对象复制到另一部分,然后清除未被标记的内存。
  5. 分代回收思想
    • 根据对象存在的时间不同,将对象划分为不同的代,一半新创建的对象,通常只存活较短的时间,而对于长时间存活的对象,采用较长的垃圾回收周期。

使用逻辑删除原理,标记消息的存在还是已经删除。虽然这样可以有效的使用数据,不过随着时间越来越久,数据量也越来越大,我们就需要对数据进行垃圾回收(JVM

  1. 使用复制算法来解决这个问题,复制算法有效的前提是文件中有效数据较少,无效数据多,将有效数据复制到一个新文件中。

  2. 何时触发,有效数据不多时,进行复制算法

大量信息时的解决方案:如果某个队列中,消息特别多,而且都是有效消息,可能就会导致这个消息数据文件特别大,之后针对这个文件进行的操作就会比较慢。

  1. 文件拆分:当单个文件达到一定阈值时,就会将这个文件拆分为两个文件,慢慢的,就会形成许多文件。
  2. 文件合并:每个单独的文件都会进行GC,如果检查后,发现这个文件变小了许多,就可能会和相邻的文件进行合并操作,节省空间。

创建文件管理类——MessageFileManger

创建统计文件数据和文件

第一步,只需要创建一个内部类囊括了统计文件数据和三个方法:

内部类:封装统计文件的两个数据,总消息数和有效消息数。

  1. 获取队列文件位置
  2. 获取统计消息文件位置
  3. 获取消息数据文件位置

具体实现代码:

public class MessageFileManger {
    //定义内部类来表示队列的统计信息
    static public class State{
        public int totalCount;//总消息数
        public int validCount;//有效消息数
    }
    public void init(){
	//初始化,扩展!
    }
    //创建两个文件
    //1.获取文件目录
    private String getQueueDir(String queueName){
        return "./data/"+queueName;
    }
    //2.创建统计文件
    private String getQueueStatePath(String queueName){
        return getQueueDir(queueName)+"/queue_state.txt";
    }
    //3.创建消息文件
    private String getQueueDataPath(String queueName){
        return getQueueDir(queueName)+"/queue_data.txt";
    }
}
统计文件的读写操作

因为统计文件是由二进制文件构成,所以这里可以直接使用inputStreamOutputStream来进行文件读写操作。在之前的基础上操作文件数据。

读取文件:

//读取统计消息文件
//读取统计消息文件
private Stat readStat(String queueName){
    //由于当前消息统计文件是文本文件。可以直接使用Scanner读取文件
    Stat stat = new Stat();
    try(InputStream inputStream = new FileInputStream(getQueueDataPath(queueName))) {
        Scanner scanner = new Scanner(inputStream);
        stat.totalCount = scanner.nextInt();
        stat.validCount = scanner.nextInt();
        return stat;
    } catch (IOException e) {
        e.printStackTrace();
    }
    return null;
}

写入文件操作:

//写入统计消息文件
private void writeStat(String queueName,Stat stat){
    //使用 printWrite来写入文件
    //使用OutPutStream会将原文件清空,相对于新数据会覆盖旧数据
    try(OutputStream outputStream = new FileOutputStream(getQueueDataPath(queueName),true)){
        //参数`true`意思是在重新写入文件时将消息写在文件末尾
        PrintWriter writer = new PrintWriter(outputStream);
        writer.write(stat.totalCount+"\t"+stat.validCount);
        writer.flush();//手动提交
    }catch (IOException e) {
        e.printStackTrace();
    }
}
创建消息对应的文件和目录
创建一个统一处理异常

在创建文件目录时,创建可能会失败,而我们不能再依靠原来的方案来进行异常抛出,引入一个新方法来统一将异常进行处理。

自定义异常:

public class MqException extends Exception{
    public MqException(String reason){
        super(reason);
    }
}

接着我们便继续创建每个消息队列专属的文件目录和文件:

//1, 创建每个消息队列专属的文件目录及文件
public void createQueueFiles(String queueName) throws IOException, MqException {
    //1. 创建队列以及对应的消息目录
    File baseDir = new File(getQueueDir(queueName));
    if(!baseDir.exists()){
        boolean ok = baseDir.mkdirs();
        if(!ok){
            throw new MqException("[MessageFileManger] 创建目录失败! queueName = "+queueName);
        }
    }
    //2. 创建数据文件
    File dataFile = new File(getQueueDataPath(queueName));
    if(!dataFile.exists()){
        boolean ok = dataFile.createNewFile();
        if(!ok){
            throw new MqException("[MessageFileManger] 创建数据文件失败! queueName = "+queueName);
        }
    }
    //3.创建统计文件
    File stateFile = new File(getQueueStatePath(queueName));
    if(!stateFile.exists()){
        boolean ok = stateFile.createNewFile();
        if(!ok){
            throw new MqException("[MessageFileManger] 创建数据文件失败! queueName = "+queueName);
        }
    }
    //4. 初始化统计文件 ----> 0/t0
    State state = new State();
    state.validCount = 0;
    state.totalCount = 0;
    writeState(queueName,state);
}

有创建便有删除操作:

//2. 删除每个消息队列专属的目录和文件
public void deleteQueueFiles(String queueName) throws MqException {
    //1.先将文件删除,再删除目录
    File stateFile = new File(getQueueStatePath(queueName));
    boolean state = stateFile.delete();
    File dataFile = new File(getQueueDataPath(queueName));
    boolean data = dataFile.delete();
    File baseFile = new File(getQueueDir(queueName));
    boolean delDir = baseFile.delete();
    if(!state || !data || !delDir){
        throw new MqException("[MessageFileManger] 删除目录失败! queueName = "+queueName);
    }
}

当需要操作一个消息队列时,我们必须提前检查消息队列的文件是否正常:

//3. 检查目录和文件是否存在,如果要对一个消息队列进行操作,就应该先判定
    public boolean checkFileExits(String queueName){
        //实际上只需要判断两个文件的存在即可,因为只要文件存在,目录便一定存在
        File dataFile = new File(getQueueDataPath(queueName));
        if(!dataFile.exists()){
            return false;
        }
        File stateFile = new File(getQueueStatePath(queueName));
        if(!stateFile.exists()){
            return false;
        }
        return true;
    }

消息文件的读写

消息的序列化和反序列化

在整理消息之前,不要忘了数据文件中存储的是二进制数据!对于二进制数据,就必须要进行序列化和反序列化

序列化的好处

将一个结构体转化成一个字符串或者字节数组存储在系统中,序列化后方便存储。

经过这个的这样的序列化后,结构体中的信息是不会丢失的,这样后面可以进行反序列化取出原来的数据。

使用二进制的序列化方式,针对Message对象进行序列化,如果一个对象能够反序列化或序列化,需要让这个类实现 Serializable 接口。

在这里插入图片描述

在公共类文件夹中创建一个序列化和反序列化的文件。

import java.io.*;
//这个类中的逻辑,不仅仅是针对消息队列的反序列化和序列化操作,对于java任意数据都通过以下逻辑进行序列化操作
// 如果一个对象能够反序列化或序列化,需要让这个类实现 Serializable 接口

public class BinaryTool {
    //把一个对象序列化成一个数组
    public static byte[] toBytes(Object object){
        //这个流对象相对于一个变长的字节数组
        // 可以把 object 序列化的数据给写入到 byteArrayOutputStream ,再统一转成 byte[]
        try(ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream()){
            try(ObjectOutputStream outputStream = new ObjectOutputStream(byteArrayOutputStream)) {
                // 此处的 writeObject 就会把该对象进行序列化,生成二进制的字节数据,就会写到 objectOutPutStream
                // 由于 objectOutPutStream 关联了 ByteArrayOutPutStream ,最终结果写入到 byteArray
                outputStream .writeObject(object);
                //这个操作就是把 byteArrayOutputStream 中持有的二进制数据取出来,转成byte[]
                return byteArrayOutputStream.toByteArray();
            }
        } catch (IOException e) {
            e.printStackTrace();
        }
        return null;
    }
    //把一个数组序列化成一个对象
    public static Object fromBytes(byte[] data) {
        Object ret = null;
        try (ByteArrayInputStream byteArrayInputStream = new ByteArrayInputStream(data)) {
            try (ObjectInputStream inputStream = new ObjectInputStream(byteArrayInputStream)) {
                //这里的readObject是在数组中读取数据并进行反序列化
                ret = inputStream.readObject();
            } catch (ClassNotFoundException e) {
                e.printStackTrace();
            }
        } catch (IOException e) {
            e.printStackTrace();
        }
        return ret;
    }
}

注意使用序列化的对象需要实现serializable接口。

实现完接口后,在对项目的版本号进行更新操作。

//验证开发者版本号
private static final long serialVersionUID=1L;

发送信息和获取信息

准备好消息队列的文件后,接下来就是想办法将消息存储到队列上,这里整理了对消息存储到文件的几个步骤。

使用两个方法来进行发送和获取:

//消息的写入与读取
//1.在创建消息序列化和反序列化的基础上,创建发送消息方法
public void sendMessage(MSGQueue queue, Message message) throws MqException, IOException {
    //1. 检查当前队列是否存在
    if(!checkFileExits(queue.getName())){
        throw new MqException("[MessageFileManger] 队列不存在!无法发送消息到队列 queueName = "+queue.getName());
    }
    //2.对将要发送的消息进行序列化处理
    byte[] messageBinary = BinaryTool.toBytes(message);
    //发送消息时可能存在线程安全问题,需要进行加锁
    synchronized (queue){
        //3. 获取队列中数据文件的长度,用来计算 Message 中的偏移量 offsetBeg 和 offsetEnd
        // 写入数据到队列的末尾 因为前面的len存储数据长度用了4个字节,所以 offsetEnd和offsetBeg都需要经过计算后+4
        // 3.1 得到数据文件
        File datafile = new File(getQueueDataPath(queue.getName()));
        // 3.2 设置偏移量,预留出消息位置
        message.setOffsetBeg(datafile.length()+4);
        message.setOffsetEnd(datafile.length()+4 + messageBinary.length);
        //3.3 追加数据到文件中
        try(OutputStream outputStream = new FileOutputStream(datafile,true)){
            try(DataOutputStream dataOutputStream = new DataOutputStream(outputStream)){
                //写入文件长度,前4个字节
                dataOutputStream.writeInt(messageBinary.length);
                //写入消息数据
                dataOutputStream.write(messageBinary);
            }
        }
        //更新消息统计文件
        State state = readState(queue.getName());
        state.totalCount +=1;
        state.validCount +=1;
        writeState(queue.getName(),state);
    }
}

写入文件中存在着线程安全问题,在两个线程同时执行这段代码时会出现特别要注意在发送消息时可能会发生线程安全问题——脏读问题。如果多个线程对一个消息队列发送消息,可能会进行消息覆盖,所以我们这里需要加锁操作。

解决方法:

以队列对象为加锁单位即可

如果是两个线程,往一个队列中写消息,此时需要阻塞等待。如果是两个线程往不同队列中写消息不需要阻塞等待。

逻辑删除信息

删除步骤

这里是逻辑删除信息,也就是将硬盘上存储的数据的 isValid 属性改为 0
1. 先把文件中数据读取出来,还原为Message对象
2. 把isValid属性改为 0
3. 在把信息重新写入文件中

这里的message参数必须包含有效的 offsetBegoffsetEnd 来标识信息的位置

此处读取信息不能使用FileOutputStreamFileInputStream,这两个类是从文件头进行读写,这里我们需要指定位置进行读取信息,这里使用RandomAccessFile,内存支持随机访问,只有这样才能更准确定位到消息。

具体代码实现:

//逻辑删除消息
    // 这里我们只需要将文章中的 valid 改成 0 即可
    public void deleteMessage(MSGQueue queue,Message message) throws IOException, ClassNotFoundException {
        //要知道删除的是哪一个队列和那条消息
        //注意!这里也需要加锁操作,因为 不能并发操作一个消息的删除工作,例如一个消息抢先一步删除,后续程序可能会出现异常情况
        synchronized (queue) {
            //设置可读写权限
            try (RandomAccessFile randomAccessFile = new RandomAccessFile(getQueueDataPath(queue.getName()), "rw")) {
                //1. 得到具体的消息
                byte[] bufferSrc = new byte[(int) (message.getOffsetEnd() - message.getOffsetBeg())];
                randomAccessFile.seek(message.getOffsetBeg());
                randomAccessFile.read(bufferSrc);
                //2. 将原来的二进制数据转为 Message 对象
                Message diskMessage = (Message) BinaryTool.fromBytes(bufferSrc);
                //3. 修改 valid 参数
                diskMessage.setIsValid((byte) 0x0);
                //4.重新覆盖当时的文件
                byte[] buffDest = BinaryTool.toBytes(diskMessage);
                // 5. 重新调整文件位置
                randomAccessFile.seek(message.getOffsetBeg());
                randomAccessFile.write(buffDest);
            }
            //因为删除了一条消息,所以统计文件也应该更新
            State state = readState(queue.getName());
            if(state.validCount > 0){
                state.validCount -= 1;
            }
            writeState(queue.getName(),state);
        }
    }

读取文件消息加载到内存中

将文件中的消息加载到内存中,当然是读取队列中的有效消息。

这个方法是在程序准备启动时执行的,计算机需要将硬盘中的有效消息加载到内存中。

这里我们使用链表这种数据结构,主要是为了后期删除时可以从头部进行删除,避免浪费更多的空间。

  1. 得到消息文件。
  2. 读取文件,记录文件开始光标,得到文件长度,计算文件结尾光标位置。将这一段封装为一个消息进行保存。在消息存储在硬盘上时进行了序列化加密操作,所以在取出消息时,也应该对消息进行反序列化操作。
  3. 判断消息是否为有效消息,如果有效则加入链表中。
  4. 循环进行扫描该消息队列。
//加载有效消息
public LinkedList<Message> loadAllMangeFromQueue(String queueName) throws IOException, MqException{
    //返回参数
    LinkedList<Message> messages = new LinkedList<>();
    //1. 得到消息文件
    try(InputStream inputStream = new FileInputStream(getQueueDataPath(queueName))){
        try(DataInputStream dataInputStream = new DataInputStream(inputStream)){
            //记录文件光标,慢慢向后移动,相对于得到下一个文件的开始光标
            long currentOffSet = 0;
            //循环查找
            while (true){
                //1.得到消息总长度
                int messageSize = dataInputStream.readInt();
                //2. 按照消息总长度+光标得到消息的位置
                byte[] buffer = new byte[messageSize];
                int actualSize = dataInputStream.read(buffer);
                //对比读取消息的实际长度和消息总长度
                if(messageSize != actualSize){
                    throw new MqException("[MessageFileManger] 消息读取错误,消息长度不符合规范!queueName ="+queueName);
                }
                //3.将数据封装为一个消息
                Message message =  (Message)BinaryTool.fromBytes(buffer);
                //4.判断数据的有效性
                if(message.getIsValid() != 0x1){
                    currentOffSet += (4+messageSize);
                    //直接跳过,无效数据
                    continue;
                }
                //5.有效数据加入链表中,注意设置消息的 OffsetBeg 和 OffSetEnd 找到在文件中具体位置
                message.setOffsetBeg(currentOffSet+4);
                message.setOffsetEnd(currentOffSet+4+messageSize);
                currentOffSet += (4+messageSize);
                messages.add(message);
            }
        }catch (EOFException | ClassNotFoundException e){
            System.out.println("[MessageFileManger] 恢复Message数据完成!");
        }
    }
    return messages;
}

垃圾回收—》复制算法

这里并不像逻辑删除那样,这个删除的数据是已经经过消费者进行使用了,所以在这里是真正的将Message进行删除。因为逻辑删除存在一系列的缺点,比如无效消息过多导致资源浪费,以及在查找消息时需要更多的时间来排除无效消息。需要注意这里并不需要将isvalid设置为0x0直接在硬盘以及内存中将文件内容进行修改即可

  1. 使用复制算法进行垃圾回收。
  2. 判断文件是否需要进行垃圾回收,根据统计文件中的有效消息和总消息,如果总消息大于2000且有效消息小于总消息的一半,那么就进行垃圾回收。
  3. 首先创建一个新文件。
  4. 在旧文件中得到有效消息然后写入新文件中,按照一个消息的格式写入。
  5. 删除旧文件,将新文件命名为旧文件名。
  6. 更新统计文件数据。
//检查是否需要进行垃圾回收
public boolean checkGC(String queueName){
    State state = readState(queueName);
    assert state != null;
    return state.totalCount > 2000 && state.validCount *2 < state.totalCount;
}
//创建一个临时文件作为复制目的
private String getQueueDataNewPath(String queueName){
    return getQueueDataPath(queueName)+"/queue_data_new.txt";
}
//复制算法实现
//写入文件!写入文件操作是存在线程安全问题的,当两个线程对同一个消息队列进行写入时可能会出现错误,所以我们需要针对队列进行加锁
public void gc(MSGQueue queue) throws MqException, IOException {
    synchronized (queue) {
        //1. 创建一个新文件
        File queueDataNewFile = new File(getQueueDataNewPath(queue.getName()));
        if (queueDataNewFile.exists()) {
            //判断新文件的存在性,注意此刻并没有进行创建操作
            throw new MqException("[MessageFileManger] gc失败,新文件已经存在! queueName = " + queue.getName());
        }
        boolean ok = queueDataNewFile.createNewFile();
        if (!ok) {
            throw new MqException("[MessageFileManger] gc时文件创建失败! queueName = " + queue.getName());
        }
        //2.读取旧文件中的有效对象
        LinkedList<Message> messages = loadAllMangeFromQueue(queue.getName());
        //3. 把有效消息写入新文件
        try (OutputStream outputStream = new FileOutputStream(queueDataNewFile)) {
            try (DataOutputStream dataInputStream = new DataOutputStream(outputStream)) {
                for (Message m : messages) {
                    byte[] buffer = BinaryTool.toBytes(m);
                    //按照格式写入新文件
                    dataInputStream.writeInt(buffer.length);
                    dataInputStream.write(buffer);
                }
            }
        }
        //4.删除旧文件,将新文件重新命名
        File queueOldFile = new File(getQueueDataPath(queue.getName()));
        ok = queueOldFile.delete();
        if (!ok) {
            throw new MqException("[MessageFileManger] gc时旧文件删除失败! queueName = " + queue.getName());
        }
        ok = queueDataNewFile.renameTo(queueOldFile);
        if (!ok) {
            throw new MqException("[MessageFileManger] gc时新文件修改文件名失败! queueName = " + queue.getName());
        }
        //5.更新统计文件
        State state = readState(queue.getName());
        state.validCount = messages.size();
        state.totalCount = messages.size();
        writeState(queue.getName(), state);
        System.out.println("[MessageFileManger] gc执行完毕! queueName = " + queue.getName());
    }
}
MessageFileManger总结:
  1. 设计了目录结构和文件格式
  2. 实现了目录创建和删除
  3. 实现了统计文件的读写
  4. 实现了消息的写入,写入数据文件
  5. 消息删除 ==》随机访问文件
  6. 加载所有消息
  7. 垃圾回收机制===》复制算法

测试功能

测试文件管理功能,每写完一个功能都需要进行一个单元测试,为了避免后续操作的错误,我们应该将当前功能测试完毕再进行后续文件的编码。

文件操作测试用例:

  1. 做好每个用例的准备工作
    • 创建队列以备后续使用
  2. 做好收尾工作
    • 释放准备工作中创建的资源

具体实现:

@SpringBootTest
public class MessageFileMangeTest {
    private MessageFileManger messageFileManger ;
    private static final String queueName1 = "testQueue1";
    private static final String queueName2 = "testQueue2";
    @BeforeEach
    public void setUp() throws IOException, MqException {
        messageFileManger = new MessageFileManger();
        messageFileManger.createQueueFiles(queueName1);
        messageFileManger.createQueueFiles(queueName2);
    }
    @AfterEach
    public void tearDown() throws MqException {
        messageFileManger.deleteQueueFiles(queueName1);
        messageFileManger.deleteQueueFiles(queueName2);
    }
}

功能一:创建消息文件和统计文件

//1.测试文件是否可以被创建
@Test
public void testCreateFile(){
    File queueDataFile1 = new File("./Data/" + queueName1 + "/queue_data.txt");
    Assertions.assertEquals(true,queueDataFile1.isFile());
    File queueStateFile1 = new File("./data/"+ queueName1+"/queue_state.txt");
    Assertions.assertEquals(true,queueStateFile1.isFile());

    File queueDataFile2 = new File("./Data/" + queueName2 + "/queue_data.txt");
    Assertions.assertEquals(true,queueDataFile2.isFile());
    File queueStateFile2 = new File("./data/"+ queueName2+"/queue_state.txt");
    Assertions.assertEquals(true,queueStateFile2.isFile());
}

测试结果:

在这里插入图片描述

功能二:统计文件的正常写入和读取操作(利用反射使用 private方法)

@Test
public void testReadWriteState(){
    MessageFileManger.State state = new MessageFileManger.State();
    state.totalCount = 1000;
    state.validCount = 600;
    //使用反射调用 private 方法
    ReflectionTestUtils.invokeMethod(messageFileManger,"writeState",queueName1,state);
    //写入state文件后读取
    MessageFileManger.State newState = ReflectionTestUtils.invokeMethod(messageFileManger,"readState",queueName1);
    Assertions.assertEquals(1000,newState.totalCount);
    Assertions.assertEquals(600,newState.validCount);
}

在这里插入图片描述

功能三:测试发送数据到队列

//1创建一个队列
private MSGQueue createQueue(String queueName) {
    MSGQueue queue = new MSGQueue();
    queue.setName(queueName);
    queue.setDurable(true);//持久化
    queue.setAutoDelete(false);//不使用删除
    queue.setExclusive(false);//只属于一个消费者
    return queue;
}
//2.初始化一个消息
private Message createTestMessage(String contain) {
    return Message.createMessage("testRoutingKey", null, contain.getBytes());
}
@Test
public void testSendMessage() throws IOException, MqException {
    MSGQueue queue = createQueue(queueName1);//调用了创建队列函数
    Message message = createTestMessage("testMessage");//调用了初始化消息函数
    //调用完发送消息,下一步应该检查统计文件和消息文件的内容
    messageFileManger.sendMessage(queue,message);

    //检查State
    MessageFileManger.State state = ReflectionTestUtils.invokeMethod(messageFileManger, "readState", queueName1);
    Assertions.assertEquals(1,state.validCount);
    Assertions.assertEquals(1,state.totalCount);
    //检查消息文件
    LinkedList<Message> messages = messageFileManger.loadAllMangeFromQueue(queueName1);
    Assertions.assertEquals(1,messages.size());
    Message message1 = messages.get(0);
    Assertions.assertEquals(message.getMessageId(),message1.getMessageId());
    Assertions.assertEquals(message.getRoutingKey(),message1.getRoutingKey());
    Assertions.assertEquals(message.getDeliverMode(),message1.getDeliverMode());

    Assertions.assertArrayEquals(message.getBody(),message1.getBody());
}

在这里插入图片描述

功能四:测试在队列中读取消息

@Test
public void testLoadAllMessageFromQueue() throws IOException, MqException {
    MSGQueue queue = createQueue(queueName1);
    LinkedList<Message> exceptMessages = new LinkedList<>();//存入实际数据
    //插入数据
    for (int i = 0; i < 100; i++) {
        Message message = createTestMessage("testMessage" + i);
        messageFileManger.sendMessage(queue,message);
        exceptMessages.add(message);
    }
    //读取消息并对比
    LinkedList<Message> actualMessages = messageFileManger.loadAllMangeFromQueue(queue.getName());
    Assertions.assertEquals(exceptMessages.size(),actualMessages.size());//对比大小
    for (int i = 0; i < exceptMessages.size(); i++) {
        //对比内容
        Message exceptMessage = exceptMessages.get(i);
        Message actualMessage = actualMessages.get(i);
        System.out.println("["+i+"]actualMessages = "+actualMessage);
        Assertions.assertEquals(exceptMessage.getMessageId(),actualMessage.getMessageId());
        Assertions.assertEquals(exceptMessage.getRoutingKey(),actualMessage.getRoutingKey());
        Assertions.assertEquals(exceptMessage.getDeliverMode(),actualMessage.getDeliverMode());
        Assertions.assertEquals(0x1,actualMessage.getIsValid());
    }
}

在这里插入图片描述

功能五:测试删除消息数据

//测试删除消息功能
@Test
public void testDeleteMessage() throws IOException, MqException, ClassNotFoundException {
    MSGQueue queue = createQueue(queueName1);
    LinkedList<Message> exceptMessages = new LinkedList<>();
    for (int i = 0; i < 10; i++) {
        Message testMessage = createTestMessage("testMessage" + i);
        exceptMessages.add(testMessage);
        messageFileManger.sendMessage(queue,testMessage);
    }
    //删除三个消息
    messageFileManger.deleteMessage(queue,exceptMessages.get(7));
    messageFileManger.deleteMessage(queue,exceptMessages.get(8));
    messageFileManger.deleteMessage(queue,exceptMessages.get(9));

    //对比剩余消息
    LinkedList<Message> actualMessages = messageFileManger.loadAllMangeFromQueue(queue.getName());
    Assertions.assertEquals(7,actualMessages.size());
    for (int i = 0; i < 7 ; i++) {
        Message exceptMessage = exceptMessages.get(i);
        Message actualMessage = actualMessages.get(i);
        System.out.println("["+i+"]actualMessages = "+actualMessage);
        Assertions.assertEquals(exceptMessage.getMessageId(),actualMessage.getMessageId());
        Assertions.assertEquals(exceptMessage.getRoutingKey(),actualMessage.getRoutingKey());
        Assertions.assertEquals(exceptMessage.getDeliverMode(),actualMessage.getDeliverMode());
        Assertions.assertEquals(0x1,actualMessage.getIsValid());
    }
}

在这里插入图片描述

功能六:测试垃圾回收,真正的删除内存中的消息

//1.写入消息
//2.删除一半消息
//3.手动GC
@Test
public void testGC() throws IOException, MqException, ClassNotFoundException {
    MSGQueue queue = createQueue(queueName1);
    LinkedList<Message> exceptMessages = new LinkedList<>();
    for (int i = 0; i < 100; i++) {
        Message testMessage = createTestMessage("testMessage" + i);
        exceptMessages.add(testMessage);
        messageFileManger.sendMessage(queue,testMessage);
    }
    //获取GC前文件
    File beforeGC= new File("./data" + queueName1 + "queue_data.txt");
    long begGCLength = beforeGC.length();
    System.out.println(begGCLength);
    //删除下标为偶数的文件
    for (int i = 0; i < 100; i+=2) {
        messageFileManger.deleteMessage(queue,exceptMessages.get(i));
    }
    //手动调用垃圾回收
    messageFileManger.gc(queue);
    //重新读取消息并验证消息的准确性
    List<Message> actualMessage = messageFileManger.loadAllMangeFromQueue(queue.getName());
    System.out.println(exceptMessages.size()+"==="+actualMessage.size());
    Assertions.assertEquals(50,actualMessage.size());
    //对比两个数组之间的消息是否一致
    for (int i = 0; i < actualMessage.size(); i++) {
        Message actualMessages = actualMessage.get(i);
        Message exceptMessage = exceptMessages.get(i * 2 + 1);

        Assertions.assertEquals(exceptMessage.getMessageId(),actualMessages.getMessageId());
        Assertions.assertEquals(exceptMessage.getRoutingKey(),actualMessages.getRoutingKey());
        Assertions.assertEquals(exceptMessage.getDeliverMode(),actualMessages.getDeliverMode());
        Assertions.assertEquals(0x1,actualMessages.getIsValid());
    }
    File afterGCFile = new File("./data/" + queueName1 + "queue_data.txt");
    long aftGCLength = afterGCFile.length();
    System.out.println(aftGCLength);
    Assertions.assertTrue(begGCLength>=aftGCLength);
}

在这里插入图片描述

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

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

相关文章

算法通关村第一关——链表经典问题之合并有序链表三种方法一层一层优化

算法通关村第一关——链表经典问题之合并有序链表三种方法一层一层优化 题目描述 将两个升序的链表合并为一个新的升序链表并返回&#xff0c;新链表是通过拼接两个给定的两个链表的所有节点组成的。 解题思路 第一种 新建一个链表&#xff0c;然后分别遍历两个链表&#…

记一次edu站点并拿下的过程cnvd

0x01 jeecg-boot介绍 JeecgBoot是一款基于代码生成器的低代码开发平台&#xff0c;零代码开发&#xff01;采用前后端分离架构&#xff1a;SpringBoot2.x&#xff0c;Ant Design&Vue&#xff0c;Mybatis-plus&#xff0c;Shiro&#xff0c;JWT。强大的代码生成器让前后端代…

k8s-17 k8s调度

调度器通过 kubernetes 的 watch 机制来发现集群中新创建且尚未被调度到 Node上的 Pod。调度器会将发现的每一个未调度的 Pod 调度到一个合适的 Node 上来运行。 kube-scheduler 是 Kubernetes 集群的默认调度器&#xff0c;并且是集群控制面的一部分如果你真的希望或者有这方面…

python如何连接数据库 ?一文详解pymysql的用法 。

使用Python连接数据库是常用的操作 &#xff0c;那么在Python代码中取操作数据库呢 &#xff1f; 接下来介绍一个包 &#xff1a;pymysql .它能帮我们在代码中连接MySQL数据库进行各种操作。 1.常用数据库模块 在做自动化测试时&#xff0c;我们经常会查库的需求 &#xff0c;…

解剖—顺序表相关OJ练习题

目录 一、删除有序数组中的重复项&#xff0c;返回出现一次元素的个数。 二、原地移除数组中所有数值等于val的元素 三、合并两个有序数组 四、旋转数组 五、数组形式的整数加法 一、删除有序数组中的重复项&#xff0c;返回出现一次元素的个数。 26. 删除有序数组中的重…

Service Mesh和Kubernetes:加强微服务的通信与安全性

文章目录 什么是Service Mesh&#xff1f;Service Mesh的优势1. 流量控制2. 安全性3. 可观测性 Istio&#xff1a;Service Mesh的领军者流量管理安全性可观测性 Linkerd&#xff1a;轻量级Service Mesh流量管理安全性可观测性 Istio vs. Linkerd实际应用结论 &#x1f388;个人…

vscode中4个json的区别和联系

在vscode中快捷键ctrlshiftp&#xff0c;然后输入setting&#xff0c;会出现下图几个选项 当不同设置之间出现冲突时&#xff0c;听谁的&#xff1a; Open Workspace Settings(JSON) > Open Settings(JSON) Open User Settings > Open Default Settings(JSON) Open Wo…

openstack 云主机 linux报 login incorrect

还未输入密码就提示login incorrect 不给输密码位置 完全不给输密码的机会 关机进入单用户 检查登录安全记录 vi /var/log/secure 发现 /usr/lib64/security/pam_unix.so 报错 将正常的机器提取/usr/lib64/security/pam_unix.so 比对MD5一致&#xff0c; 另外判断 libtir…

车载开发学习——CAN总线

CAN总线又称为汽车总线&#xff0c;全程为“控制器局域网&#xff08;Controller Area Network&#xff09;”&#xff0c;即区域网络控制器&#xff0c;它将区域内的单一控制单元以某种形式连接在一起&#xff0c;形成一个系统。在这个系统内&#xff0c;大家以一种大家都认可…

市值缩水90%以上,泛生子何以败退美股?

癌症是人类面临的最大健康威胁之一&#xff0c;也是医学界最难攻克的难题之一。随着科技的发展&#xff0c;癌症精准医疗逐渐成为治疗癌症的新方向&#xff0c;癌症精准医疗能通过对癌细胞的基因检测和分析&#xff0c;为患者提供个性化的治疗方案。然而&#xff0c;这一领域的…

redis(其它操作、管道)、django中使用redis(通用方案、 第三方模块)、django缓存、celery介绍(celery的快速使用)

1 redis其它操作 2 redis管道 3 django中使用redis 3.1 通用方案 3.2 第三方模块 4 django缓存 5 celery介绍 5.1 celery的快速使用 1 redis其它操作 delete(*names) exists(name) keys(pattern*) expire(name ,time) rename(src, dst) move(name, db)) randomkey() type(na…

VBA技术资料MF71:查找所有空格并替换为固定字符

我给VBA的定义&#xff1a;VBA是个人小型自动化处理的有效工具。利用好了&#xff0c;可以大大提高自己的工作效率&#xff0c;而且可以提高数据的准确度。我的教程一共九套&#xff0c;分为初级、中级、高级三大部分。是对VBA的系统讲解&#xff0c;从简单的入门&#xff0c;到…

实现实时美颜:主播直播美颜SDK的技术细节

在今天的数字时代&#xff0c;直播和实时互动成为了日常生活的一部分&#xff0c;而主播直播美颜SDK的出现为用户提供了更加精美的视觉体验。这项技术的背后有着复杂的技术细节&#xff0c;从图像处理到机器学习&#xff0c;本文将深入探讨主播直播美颜SDK的技术细节&#xff0…

四边形不等式

区间dp问题&#xff0c;状态转移方程&#xff1a; dp[i][j] min( dp[i][k] dp[k1][j] w[i][j] ) //w[i][j]是从i到j的&#xff0c;一个定值 不随k改变&#xff0c;而且w的值只和i j有关&#xff0c;是它们的二元函数。 其中i<k<j ,初始值dp[i][i]已知。 含义&#x…

第三类医疗器械经营许可证经营范围

在我国&#xff0c;医疗器械监督管理条例规定:医械经营企业要依据主营产品办理相应许可证。医疗器械根据其风险性又分为三类&#xff0c;一类医疗器械实行产品备案管理&#xff0c;第二类、第三类医疗器械实行产品注册管理&#xff0c;经营第二类、第三类医疗器械应当持有《医疗…

Day 2 Qt

#include "my_widget.h" #include "ui_my_widget.h"My_Widget::My_Widget(QWidget *parent): QWidget(parent), ui(new Ui::My_Widget) {ui->setupUi(this);//窗口的相关设置 // this -> resize(800,500);this -> setWindowTitle("QQ聊天…

APP备案公钥、证书MD5指纹/签名MD5值获取方法

本文只详细讲解android app获取方法&#xff0c;三种方式&#xff1a; 1. 你的应用已安装到手机&#xff0c;android应用市场搜索下载安装 APP备案助手&#xff0c;此app可直接获取所有已安装app的公钥、证书MD5指纹/签名MD5值&#xff0c;示例&#xff1a;获取 抖音app公钥、…

每日一练 | 华为认证真题练习Day120

1、MPLS域中的LER全称为Label Egress Router。 A. 对 B. 错 2、如果一个以太网数据帧的Type/Length字段的值为0x0800&#xff0c;则此数据帧所承载的上层报文首部长度范围为20-60B。 A. 对 B. 错 3、在VRP平台上&#xff0c;可以通过下面哪种方式访问上一条历史命令&#x…

AI爆文变现-写作项目-脚本配置教程-解放双手

之前给大家分享过AI爆文的写作教程&#xff0c;没看过的可以看下对应的教程&#xff1a; AI爆文撸流量主保姆级教程2.0 因为是怼量&#xff0c;为了高效完成文章&#xff0c;我用python脚本实现了自动写文章的功能&#xff0c;发布文章目前还是要手动进行。 AI爆文教程参考&…

C++11——包装器与lambda表达式

目录 一.背景 二.lambda 1.见一见lambda 2.lambda表达式语法 3.lambda捕捉列表说明 三.函数对象与lambda表达式 四.包装器 1.function包装器 2.包装类的成员函数 五.bind 1.调整参数位置 2.减少函数参数 一.背景 在C98中&#xff0c;如果想要对一个数据集合中的元素…