项目中大量的使用 NIO 和 Mina,虽然之前一直对这部分比较关注,但是还没有好好的总结一下这方面的内容,本节内容介绍一下NIO 里最基本的一个类 ByteBuffer。
关于Mina中的 IoBuffer,我们可以先看 Mina API 中的描述:
A byte buffer used by MINA applications. This is a replacement for ByteBuffer. Please refer to ByteBuffer documentation for preliminary usage。
当然,接下去也有写到:MINA does not use NIO ByteBuffer directly for two reasons,至于这 Two Reasons,我们将在后面的比较中展开。
ByteBuffer 继承了 Buffer,对 Buffer 的理解可以说是 NIO 的入门。
在 Buffer 中有 4 个重要的属性:
它们的关系如下:
0<=mark<=position<=limit<=capacity
通俗的讲:
flip() | 读写模式的转换。 |
rewind() | 将 position 重置为 0 ,一般用于重复读。 |
clear() | 清空 buffer ,准备再次被写入 (position 变成 0 , limit 变成 capacity) 。 |
compact() | 将未读取的数据拷贝到 buffer 的头部位。 |
mark() | 可以标记一个位置 |
reset() | 可以重置到该位置。 |
get()、getShort() | 获取 ByteBuffer 中的内容,当然这里 get 的内容都是从 position 开始的,所以要时刻注意 position。每次 get 之后 position 都会改变。Position 的变化是根据你 get 的类型,如果是 short,那就是 2 个 byte,如果是 int,那就是增加 4 个 byte,即 32。 |
put()、putShort() | 向 ByteBuffer 添加内容,这里 put 的内容都是从 position 开始的。每次 put 之后 position 都会改变。 |
当然还有 allocate、hasRemaining 等常用的方法,不过这些用法一般都不会出错,使用起来和 4 个 attributes 也没有多大相关。
特别注意:Buffers are not thread-safe. If you want to access a given buffer concurrently from multiple threads, you will need to do your own synchronization prior to accessing the buffer. 至于 Buffer 或者 ByteBuffer 有什么用?那太多了,只要涉及到传输、涉及到通信,都可以用到。当然你也可以用它最原始的含义,缓冲。
package com.hl.magic.mina3; import ch.qos.logback.core.encoder.ByteArrayUtil; import org.junit.jupiter.api.Test; import java.nio.ByteBuffer; /** * @Author HL * @Date 2022/7/6 20:49 */ public class ByteBufferDemo { private final byte[] byteValue = new byte[]{0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x10}; @Test public void byteBufferWrapTest() { System.out.println("origin Data is : [" + ByteArrayUtil.toHexString(byteValue) + "]"); // 将byte数组放在buffer缓冲区 ByteBuffer buffer = ByteBuffer.wrap(byteValue); // 取出头部两个字节的buffer数组 byte[] result = new byte[2]; buffer.get(result); System.out.println("result is : [" + ByteArrayUtil.toHexString(result) + "]"); //取出body部分 byte[] body = new byte[buffer.limit() - 2]; buffer.get(body); buffer.clear(); System.out.println("body is : [" + ByteArrayUtil.toHexString(body) + "]"); //另一种取法 buffer = ByteBuffer.wrap(byteValue); body = new byte[buffer.limit() - 4]; buffer.position(4); buffer.get(body); buffer.clear(); System.out.println("body is : [" +ByteArrayUtil.toHexString(body) + "]"); } @Test public void bufferAllocateTest() { // 设置一个buffer缓冲区间,区间容量为16字节 ByteBuffer buffer = ByteBuffer.allocate(16); // 向缓冲区间添加数据 buffer.put(new byte[]{0x07, 0x08, 0x09, 0x10}); buffer.put(new byte[]{0x00, 0x00, 0x00, 0x00}); buffer.put(new byte[]{0x01, 0x01, 0x01, 0x01}); buffer.put(new byte[]{0x02, 0x02, 0x02, 0x02}); buffer.flip(); //创建一个字节数组,数组长度为buffer区间容量 byte[] bytes = new byte[buffer.limit()]; // 取出所有数据 buffer.get(bytes); buffer.clear(); System.out.println("body is : [" +ByteArrayUtil.toHexString(bytes) + "]"); //创建一个新数组,长度为buffer容量- 12字节 byte[] body = new byte[buffer.limit() - 12]; //从新的偏移位置处计算,也就是偏移12字节的区间后开始取数据 buffer.position(12); buffer.get(body); buffer.clear(); System.out.println("body is : [" + ByteArrayUtil.toHexString(body) + "]"); } }
输出:
origin Data is : [01020304050607080910] #用例1: result is : [0102] body is : [0304050607080910] body is : [050607080910] #用例2: body is : [07080910000000000101010102020202] body is : [02020202]
好了 NIO 的 ByteBuffer 告一段落,接下来先说 IoBuffer 中说不用 ByteBuffer 的 Two Reasons:
1. It doesn't provide useful getters and putters such as fill, get/putString, and get/putAsciiInt() enough.
2. It is difficult to write variable-length data due to its fixed capacity
IoBuffer buf = IoBuffer.allocate(1024).setAutoExpand(true);
在 IoBuffer 的源码中,大部分都使用了原生的 ByteBuffer 来实现,这部分采用allocator 来实现。
/** The allocator used to create new buffers */ private static IoBufferAllocator allocator = new SimpleBufferAllocator();
在 SimpleBufferAllocator 的其中一段 allocate:
public ByteBuffer allocateNioBuffer(int capacity, boolean direct) { ByteBuffer nioBuffer; if (direct) { nioBuffer = ByteBuffer.allocateDirect(capacity); } else { nioBuffer = ByteBuffer.allocate(capacity); } return nioBuffer; }
至于其他操作和 ByteBuffer 类似。
贴一段 Reilly - Java NIO 中的一段代码作为结束,后续更多的 Mina 相关会根据开发进度给出介绍:
import java.nio.CharBuffer; public class BufferFillDrain { private static int index = 0; private static String[] strings = { "A random string value", "The product of an infinite number of monkeys", "Hey hey we're the Monkees" }; private static boolean fillBuffer(CharBuffer buffer) { if (index >= strings.length) { return false; } String string = strings[index++]; for (int i = 0; i < string.length(); i++) { buffer.put(string.charAt(i)); } return true; } private static void drainBuffer(CharBuffer buffer){ while(buffer.hasRemaining()){ System.out.println(buffer.get()); } System.err.println(""); } public static void main(String[] args) { CharBuffer buffer=CharBuffer.allocate(100); while(fillBuffer(buffer)){ buffer.flip(); drainBuffer(buffer); buffer.clear(); } } }