Передача байтов из одного ByteBuffer в другой
Какой самый эффективный способ поместить как можно больше байтов из ByteBuffer bbuf_src
в другой ByteBuffer bbuf_dest
(а также узнать, сколько байтов было передано)? я пытаюсь bbuf_dest.put(bbuf_src)
но, похоже, он хочет вызвать исключение BufferOverflowException, и я не могу получить javadocs от Sun сейчас (проблемы с сетью), когда они мне нужны. >:(Ммм.
edit: darnit, подход @ Ричарда (используйте put() из резервного массива bbuf_src
) не будет работать, если bbuf_src является буфером ReadOnly, так как вы не можете получить доступ к этому массиву. Что я могу сделать в этом случае???
3 ответа
Хорошо, я адаптировал ответ @ Ричарда:
public static int transferAsMuchAsPossible(
ByteBuffer bbuf_dest, ByteBuffer bbuf_src)
{
int nTransfer = Math.min(bbuf_dest.remaining(), bbuf_src.remaining());
if (nTransfer > 0)
{
bbuf_dest.put(bbuf_src.array(),
bbuf_src.arrayOffset()+bbuf_src.position(),
nTransfer);
bbuf_src.position(bbuf_src.position()+nTransfer);
}
return nTransfer;
}
и тест, чтобы убедиться, что он работает:
public static boolean transferTest()
{
ByteBuffer bb1 = ByteBuffer.allocate(256);
ByteBuffer bb2 = ByteBuffer.allocate(50);
for (int i = 0; i < 100; ++i)
{
bb1.put((byte)i);
}
bb1.flip();
bb1.position(5);
ByteBuffer bb1a = bb1.slice();
bb1a.position(2);
// bb3 includes the 5-100 range
bb2.put((byte)77);
// something to see this works when bb2 isn't empty
int n = transferAsMuchAsPossible(bb2, bb1a);
boolean itWorked = (n == 49);
if (bb1a.position() != 51)
itWorked = false;
if (bb2.position() != 50)
itWorked = false;
bb2.rewind();
if (bb2.get() != 77)
itWorked = false;
for (int i = 0; i < 49; ++i)
{
if (bb2.get() != i+7)
{
itWorked = false;
break;
}
}
return itWorked;
}
Как вы обнаружили, получение резервного массива не всегда работает (он не работает для буферов только для чтения, прямых буферов и буферов файлов с отображением в памяти). Лучшей альтернативой является дублирование исходного буфера и установка нового ограничения на объем данных, которые вы хотите передать:
int maxTransfer = Math.min(bbuf_dest.remaining(), bbuf_src.remaining());
// use a duplicated buffer so we don't disrupt the limit of the original buffer
ByteBuffer bbuf_tmp = bbuf_src.duplicate ();
bbuf_tmp.limit (bbuf_tmp.position() + maxTransfer);
bbuf_dest.put (bbuf_tmp);
// now discard the data we've copied from the original source (optional)
bbuf_src.position(bbuf_src.position() + maxTransfer);
Вы получаете BufferOverflowException, потому что ваш bbuf_dest недостаточно велик.
Вам нужно будет использовать bbuf_dest.remaining(), чтобы узнать максимальное количество байтов, которое вы можете передать из bbuf_src:
int maxTransfer = Math.min(bbuf_dest.remaining(), bbuf_src.remaining());
bbuf_dest.put(bbuf_src.array(), 0, maxTransfer);