ByteBuf 扩容机制是指在写入数据时,如果当前容量不足以容纳新增的数据,则需要进行动态扩容,以适应数据量的增长。
下面是ByteBuf 扩容机制的详细阐述:
总的来说,ByteBuf 的扩容机制主要包括容量检查、内存分配、数据迁移、索引更新和内存释放等步骤。这个机制确保了 ByteBuf 在写入数据时能够动态地适应数据量的变化,从而保证了其灵活性和高效性。
这段代码是 ByteBuf 接口中的一个方法声明,表示向缓冲区中写入一个字节,并将写入位置的索引增加 1。
/**
* 向当前 {@code writerIndex} 处设置指定的字节,并将 {@code writerIndex} 在缓冲区中增加 {@code 1}。
* 指定值的高 24 位将被忽略。
* 如果 {@code this.writableBytes} 小于 {@code 1},则将调用 {@link #ensureWritable(int)},
* 尝试扩展容量以容纳。
*/
public abstract ByteBuf writeByte(int value);
这个方法用于向缓冲区中写入一个字节,参数 value 表示要写入的字节值。如果当前可写入的字节数小于 1(即缓冲区容量不足以容纳新的字节),则会调用 ensureWritable(int)
方法来尝试扩展缓冲区的容量,以确保能够容纳新的字节。
实现了 ByteBuf 接口中的 writeByte 方法,用于向缓冲区中写入一个字节。
@Override
public ByteBuf writeByte(int value) {
// 确保缓冲区有足够的可写空间
ensureWritable0(1);
// 将字节写入当前写入位置,并将写入位置后移一位
_setByte(writerIndex++, value);
return this;
}
该方法首先调用 ensureWritable0
方法确保缓冲区有足够的可写空间来容纳一个字节。然后调用 _setByte
方法将指定的字节值写入当前的写入位置,并将写入位置向后移动一个字节的长度。最后返回当前 ByteBuf 实例,以支持链式调用。
这段代码实现了 ensureWritable0 方法,用于确保缓冲区有足够的可写空间来容纳指定的字节数。以下是对代码的理解和注释:
final void ensureWritable0(int minWritableBytes) {
// 确保缓冲区是可访问的(未被释放)
ensureAccessible();
// 如果可写字节数大于等于要求的最小可写字节数,则无需扩容,直接返回
if (minWritableBytes <= writableBytes()) {
return;
}
// 检查是否超出最大容量限制
if (checkBounds) {
if (minWritableBytes > maxCapacity - writerIndex) {
throw new IndexOutOfBoundsException(String.format(
"writerIndex(%d) + minWritableBytes(%d) exceeds maxCapacity(%d): %s",
writerIndex, minWritableBytes, maxCapacity, this));
}
}
// 将当前容量规范化为2的幂次方,以便进行内存分配
int newCapacity = alloc().calculateNewCapacity(writerIndex + minWritableBytes, maxCapacity);
// 调整缓冲区容量为新的容量
capacity(newCapacity);
}
该方法首先确保缓冲区是可访问的,即未被释放。然后检查当前可写字节数是否满足需求,如果不满足,则计算需要扩容的容量。如果启用了边界检查(checkBounds),还会检查是否超出了最大容量限制。最后,根据计算得到的新容量,调用 capacity 方法进行容量调整。
这段代码实现了 calculateNewCapacity
方法,用于计算缓冲区扩容时的新容量。
static final int DEFAULT_MAX_CAPACITY = Integer.MAX_VALUE;
static final int CALCULATE_THRESHOLD = 1048576 * 4; // 4 MiB page
@Override
public int calculateNewCapacity(int minNewCapacity, int maxCapacity) {
// 检查最小新容量是否为正数或零
checkPositiveOrZero(minNewCapacity, "minNewCapacity");
// 如果最小新容量大于最大容量,则抛出异常
if (minNewCapacity > maxCapacity) {
throw new IllegalArgumentException(String.format(
"minNewCapacity: %d (expected: not greater than maxCapacity(%d)",
minNewCapacity, maxCapacity));
}
// 计算阈值,即4 MiB页面大小
final int threshold = CALCULATE_THRESHOLD; // 4 MiB page
// 如果最小新容量等于阈值,则返回阈值
if (minNewCapacity == threshold) {
return threshold;
}
f// 采用步进4MB的方式完成扩容
// 如果超过阈值,则不是按照两倍增长,而是按照阈值增长
if (minNewCapacity > threshold) {
int newCapacity = minNewCapacity / threshold * threshold;
if (newCapacity > maxCapacity - threshold) {
newCapacity = maxCapacity;
} else {
newCapacity += threshold;
}
return newCapacity;
}
// 采用64为基数,做倍增的方式完成扩容
// 如果未超过阈值,则按照两倍增长,直到大于等于最小新容量或者达到最大容量
int newCapacity = 64;
while (newCapacity < minNewCapacity) {
newCapacity <<= 1;
}
return Math.min(newCapacity, maxCapacity);
}
该方法首先检查最小新容量是否为正数或零,并确保不大于最大容量。然后根据阈值进行不同的扩容策略:
Netty的ByteBuf需要动态扩容来满足需要, 这种动态扩容机制通过阈值来判断采用不同的扩容策略: