【问题标题】:How do you get successive slices out of a ByteBuffer?如何从 ByteBuffer 中获取连续切片?
【发布时间】:2011-07-04 22:38:22
【问题描述】:

我有一个 ByteBuffer,其中包含一个大文件 (100 MB):

    java.nio.ByteBuffer byteBuffer = ByteBuffer.wrap(multipartFile.getBytes());
    writeChannel.write(byteBuffer);
    writeChannel.closeFinally();

我一次只能合法地向writeChannel 写入 1 MB。

如何对 ByteBuffer 的内容进行切片并一次仅将 1 MB 切片写入writeChannel

【问题讨论】:

  • 为什么?只需编写缓冲区,让系统决定如何将其打包或您要执行的任何操作。

标签: java bytebuffer


【解决方案1】:

您可以使用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循环结构。

【讨论】:

    【解决方案2】:

    据我所知,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() 中所说:“在一系列通道读取或放置操作之后,调用此方法以准备一系列通道写入或相关获取操作”

    【讨论】:

      猜你喜欢
      • 2012-08-09
      • 1970-01-01
      • 2018-07-14
      • 2022-10-14
      • 2017-08-09
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      相关资源
      最近更新 更多