我有ByteBuffer,其中包含一个大文件(100 MB):
java.nio.ByteBuffer byteBuffer = ByteBuffer.wrap(multipartFile.getBytes());
writeChannel.write(byteBuffer);
writeChannel.closeFinally();
我一次只能合法地向writeChannel
写1 MB。
如何切片ByteBuffer的内容,一次只将1 MB的切片写入writeChannel
?
答案 0 :(得分:10)
您可以使用ByteBuffer#slice()
获取基本ByteBuffer
实例的重复视图,然后沿着该位置突出显示内容的滑动窗口。或者,如果您不需要将其暴露给任何其他消费者,您可以对基本缓冲区执行相同的操作。
您可以通过the single-argument Buffer#position(int)
method更改内容视图的起始位置,并通过Buffer#limit(int)
更改视图的结束位置。只要您注意不要将视图超出底层缓冲区的限制,您就可以执行以下操作:
final ByteBuffer view = base.slice();
for (int start = base.position(), end = base.limit(), stride = 1000000;
start != end;
start = view.limit())
consume(view.position(start)
.limit(start + Math.min(end - start, stride)));
我没有测试它,但看起来正确。可以重写以避免位置的初始设置,这在这里并不是绝对必要的,但是它会导致第一次重复或更加尴尬的特殊情况处理。
我这样做是为了保留基本的 for loop 结构。
答案 1 :(得分:0)
据我所知,writeChannel上的write()(对于我认为属于SocketChannel的名称)将“尝试向通道写入r个字节,其中r是剩余的字节数缓冲区,即dst.remaining(),此时调用此方法“。 (根据this)
ByteBuffer.remaining()的描述表示此方法将“返回当前位置和限制之间的元素数量。”
所以我的猜测并不是你不能编写整个ByteBuffer,而是你的代码应该在ByteBuffer对象上调用flip()使它成为:
java.nio.ByteBuffer byteBuffer = ByteBuffer.wrap(multipartFile.getBytes());
byteBuffer.flip();
writeChannel.write(byteBuffer);
writeChannel.closeFinally();
正如它在Buffer.flip()中所说:“在一系列通道读取或放置操作之后,调用此方法来准备一系列通道写入或相对获取操作”