All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.esotericsoftware.kryo.io.ByteBufferOutput Maven / Gradle / Ivy

/* Copyright (c) 2008, Nathan Sweet
 * All rights reserved.
 * 
 * Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following
 * conditions are met:
 * 
 * - Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer.
 * - Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following
 * disclaimer in the documentation and/or other materials provided with the distribution.
 * - Neither the name of Esoteric Software nor the names of its contributors may be used to endorse or promote products derived
 * from this software without specific prior written permission.
 * 
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING,
 * BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT
 * SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
 * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */

package com.esotericsoftware.kryo.io;

import java.io.IOException;
import java.io.OutputStream;
import java.nio.ByteBuffer;
import java.nio.ByteOrder;
import java.nio.CharBuffer;
import java.nio.DoubleBuffer;
import java.nio.FloatBuffer;
import java.nio.IntBuffer;
import java.nio.LongBuffer;
import java.nio.ShortBuffer;

import com.esotericsoftware.kryo.KryoException;
import com.esotericsoftware.kryo.util.UnsafeUtil;

/** An OutputStream that buffers data in a byte array and optionally flushes to another OutputStream. Utility methods are provided
 * for efficiently writing primitive types and strings.
 * 
 * @author Roman Levenstein  */
public class ByteBufferOutput extends Output {
	protected ByteBuffer niobuffer;

	protected boolean varIntsEnabled = true;

	// Default byte order is BIG_ENDIAN to be compatible to the base class
	ByteOrder byteOrder = ByteOrder.BIG_ENDIAN;

	protected final static ByteOrder nativeOrder = ByteOrder.nativeOrder();

	/** Creates an uninitialized Output. A buffer must be set before the Output is used.
	 * @see #setBuffer(ByteBuffer, int) */
	public ByteBufferOutput () {
	}

	/** Creates a new Output for writing to a direct ByteBuffer.
	 * @param bufferSize The initial and maximum size of the buffer. An exception is thrown if this size is exceeded. */
	public ByteBufferOutput (int bufferSize) {
		this(bufferSize, bufferSize);
	}

	/** Creates a new Output for writing to a direct ByteBuffer.
	 * @param bufferSize The initial size of the buffer.
	 * @param maxBufferSize The buffer is doubled as needed until it exceeds maxBufferSize and an exception is thrown. */
	public ByteBufferOutput (int bufferSize, int maxBufferSize) {
		if (maxBufferSize < -1) throw new IllegalArgumentException("maxBufferSize cannot be < -1: " + maxBufferSize);
		this.capacity = bufferSize;
		this.maxCapacity = maxBufferSize == -1 ? Util.MAX_SAFE_ARRAY_SIZE : maxBufferSize;
		niobuffer = ByteBuffer.allocateDirect(bufferSize);
		niobuffer.order(byteOrder);
	}

	/** Creates a new Output for writing to an OutputStream. A buffer size of 4096 is used. */
	public ByteBufferOutput (OutputStream outputStream) {
		this(4096, 4096);
		if (outputStream == null) throw new IllegalArgumentException("outputStream cannot be null.");
		this.outputStream = outputStream;
	}

	/** Creates a new Output for writing to an OutputStream. */
	public ByteBufferOutput (OutputStream outputStream, int bufferSize) {
		this(bufferSize, bufferSize);
		if (outputStream == null) throw new IllegalArgumentException("outputStream cannot be null.");
		this.outputStream = outputStream;
	}

	/** Creates a new Output for writing to a ByteBuffer. */
	public ByteBufferOutput (ByteBuffer buffer) {
		setBuffer(buffer);
	}

	/** Creates a new Output for writing to a ByteBuffer.
	 * @param maxBufferSize The buffer is doubled as needed until it exceeds maxCapacity and an exception is thrown. */
	public ByteBufferOutput (ByteBuffer buffer, int maxBufferSize) {
		setBuffer(buffer, maxBufferSize);
	}

	/** Creates a direct ByteBuffer of a given size at a given address.
	 * 

* Typical usage could look like this snippet: * *

	 * // Explicitly allocate memory
	 * long bufAddress = UnsafeUtil.unsafe().allocateMemory(4096);
	 * // Create a ByteBufferOutput using the allocated memory region
	 * ByteBufferOutput buffer = new ByteBufferOutput(bufAddress, 4096);
	 * 
	 * // Do some operations on this buffer here
	 * 
	 * // Say that ByteBuffer won't be used anymore
	 * buffer.release();
	 * // Release the allocated region
	 * UnsafeUtil.unsafe().freeMemory(bufAddress);
	 * 
* * @param address starting address of a memory region pre-allocated using Unsafe.allocateMemory() * @param maxBufferSize */ public ByteBufferOutput (long address, int maxBufferSize) { niobuffer = UnsafeUtil.getDirectBufferAt(address, maxBufferSize); setBuffer(niobuffer, maxBufferSize); } /** Release a direct buffer. {@link #setBuffer(ByteBuffer, int)} should be called before next write operations can be called. * * NOTE: If Cleaner is not accessible due to SecurityManager restrictions, reflection could be used to obtain the "clean" * method and then invoke it. */ public void release () { clear(); UnsafeUtil.releaseBuffer(niobuffer); niobuffer = null; } public ByteOrder order () { return byteOrder; } public void order (ByteOrder byteOrder) { this.byteOrder = byteOrder; this.niobuffer.order(byteOrder); } public OutputStream getOutputStream () { return outputStream; } /** Sets a new OutputStream. The position and total are reset, discarding any buffered bytes. * @param outputStream May be null. */ public void setOutputStream (OutputStream outputStream) { this.outputStream = outputStream; position = 0; total = 0; } /** Sets the buffer that will be written to. maxCapacity is set to the specified buffer's capacity. * @see #setBuffer(ByteBuffer, int) */ public void setBuffer (ByteBuffer buffer) { setBuffer(buffer, buffer.capacity()); } /** Sets the buffer that will be written to. The byte order, position and capacity are set to match the specified buffer. The * total is set to 0. The {@link #setOutputStream(OutputStream) OutputStream} is set to null. * @param maxBufferSize The buffer is doubled as needed until it exceeds maxCapacity and an exception is thrown. */ public void setBuffer (ByteBuffer buffer, int maxBufferSize) { if (buffer == null) throw new IllegalArgumentException("buffer cannot be null."); if (maxBufferSize < -1) throw new IllegalArgumentException("maxBufferSize cannot be < -1: " + maxBufferSize); this.niobuffer = buffer; this.maxCapacity = maxBufferSize == -1 ? Util.MAX_SAFE_ARRAY_SIZE : maxBufferSize; byteOrder = buffer.order(); capacity = buffer.capacity(); position = buffer.position(); total = 0; outputStream = null; } /** Returns the buffer. The bytes between zero and {@link #position()} are the data that has been written. */ public ByteBuffer getByteBuffer () { niobuffer.position(position); return niobuffer; } /** Returns a new byte array containing the bytes currently in the buffer between zero and {@link #position()}. */ public byte[] toBytes () { byte[] newBuffer = new byte[position]; niobuffer.position(0); niobuffer.get(newBuffer, 0, position); return newBuffer; } /** Sets the current position in the buffer. */ public void setPosition (int position) { this.position = position; this.niobuffer.position(position); } /** Sets the position and total to zero. */ public void clear () { niobuffer.clear(); position = 0; total = 0; } /** @return true if the buffer has been resized. */ protected boolean require (int required) throws KryoException { if (capacity - position >= required) return false; if (required > maxCapacity) { niobuffer.order(byteOrder); throw new KryoException("Buffer overflow. Max capacity: " + maxCapacity + ", required: " + required); } flush(); while (capacity - position < required) { if (capacity == maxCapacity) { niobuffer.order(byteOrder); throw new KryoException("Buffer overflow. Available: " + (capacity - position) + ", required: " + required); } // Grow buffer. if (capacity == 0) capacity = 1; capacity = Math.min(capacity * 2, maxCapacity); if (capacity < 0) capacity = maxCapacity; ByteBuffer newBuffer = (niobuffer != null && !niobuffer.isDirect()) ? ByteBuffer.allocate(capacity) : ByteBuffer.allocateDirect(capacity); // Copy the whole buffer niobuffer.position(0); niobuffer.limit(position); newBuffer.put(niobuffer); newBuffer.order(niobuffer.order()); // writeVarInt & writeVarLong mess with the byte order. need to keep track of the current byte order when growing final ByteOrder currentByteOrder = byteOrder; setBuffer(newBuffer, maxCapacity); byteOrder = currentByteOrder; } return true; } // OutputStream /** Writes the buffered bytes to the underlying OutputStream, if any. */ public void flush () throws KryoException { if (outputStream == null) return; try { byte[] tmp = new byte[position]; niobuffer.position(0); niobuffer.get(tmp); niobuffer.position(0); outputStream.write(tmp, 0, position); } catch (IOException ex) { throw new KryoException(ex); } total += position; position = 0; } /** Flushes any buffered bytes and closes the underlying OutputStream, if any. */ public void close () throws KryoException { flush(); if (outputStream != null) { try { outputStream.close(); } catch (IOException ignored) { } } } /** Writes a byte. */ public void write (int value) throws KryoException { if (position == capacity) require(1); niobuffer.put((byte)value); position++; } /** Writes the bytes. Note the byte[] length is not written. */ public void write (byte[] bytes) throws KryoException { if (bytes == null) throw new IllegalArgumentException("bytes cannot be null."); writeBytes(bytes, 0, bytes.length); } /** Writes the bytes. Note the byte[] length is not written. */ public void write (byte[] bytes, int offset, int length) throws KryoException { writeBytes(bytes, offset, length); } // byte public void writeByte (byte value) throws KryoException { if (position == capacity) require(1); niobuffer.put(value); position++; } public void writeByte (int value) throws KryoException { if (position == capacity) require(1); niobuffer.put((byte)value); position++; } /** Writes the bytes. Note the byte[] length is not written. */ public void writeBytes (byte[] bytes) throws KryoException { if (bytes == null) throw new IllegalArgumentException("bytes cannot be null."); writeBytes(bytes, 0, bytes.length); } /** Writes the bytes. Note the byte[] length is not written. */ public void writeBytes (byte[] bytes, int offset, int count) throws KryoException { if (bytes == null) throw new IllegalArgumentException("bytes cannot be null."); int copyCount = Math.min(capacity - position, count); while (true) { niobuffer.put(bytes, offset, copyCount); position += copyCount; count -= copyCount; if (count == 0) return; offset += copyCount; copyCount = Math.min(capacity, count); require(copyCount); } } // int /** Writes a 4 byte int. */ public void writeInt (int value) throws KryoException { require(4); niobuffer.putInt(value); position += 4; } public int writeInt (int value, boolean optimizePositive) throws KryoException { if (!varIntsEnabled) { writeInt(value); return 4; } else return writeVarInt(value, optimizePositive); } public int writeVarInt (int val, boolean optimizePositive) throws KryoException { niobuffer.position(position); int value = val; if (!optimizePositive) value = (value << 1) ^ (value >> 31); int varInt = 0; varInt = (value & 0x7F); value >>>= 7; if (value == 0) { writeByte(varInt); return 1; } varInt |= 0x80; varInt |= ((value & 0x7F) << 8); value >>>= 7; if (value == 0) { niobuffer.order(ByteOrder.LITTLE_ENDIAN); writeInt(varInt); niobuffer.order(byteOrder); position -= 2; niobuffer.position(position); return 2; } varInt |= (0x80 << 8); varInt |= ((value & 0x7F) << 16); value >>>= 7; if (value == 0) { niobuffer.order(ByteOrder.LITTLE_ENDIAN); writeInt(varInt); niobuffer.order(byteOrder); position -= 1; niobuffer.position(position); return 3; } varInt |= (0x80 << 16); varInt |= ((value & 0x7F) << 24); value >>>= 7; if (value == 0) { niobuffer.order(ByteOrder.LITTLE_ENDIAN); writeInt(varInt); niobuffer.order(byteOrder); position -= 0; return 4; } varInt |= (0x80 << 24); long varLong = (varInt & 0xFFFFFFFFL) | (((long)value) << 32); niobuffer.order(ByteOrder.LITTLE_ENDIAN); writeLong(varLong); niobuffer.order(byteOrder); position -= 3; niobuffer.position(position); return 5; } // string /** Writes the length and string, or null. Short strings are checked and if ASCII they are written more efficiently, else they * are written as UTF8. If a string is known to be ASCII, {@link #writeAscii(String)} may be used. The string can be read using * {@link Input#readString()} or {@link Input#readStringBuilder()}. * @param value May be null. */ public void writeString (String value) throws KryoException { niobuffer.position(position); if (value == null) { writeByte(0x80); // 0 means null, bit 8 means UTF8. return; } int charCount = value.length(); if (charCount == 0) { writeByte(1 | 0x80); // 1 means empty string, bit 8 means UTF8. return; } // Detect ASCII. boolean ascii = false; if (charCount > 1 && charCount < 64) { ascii = true; for (int i = 0; i < charCount; i++) { int c = value.charAt(i); if (c > 127) { ascii = false; break; } } } if (ascii) { if (capacity - position < charCount) writeAscii_slow(value, charCount); else { byte[] tmp = value.getBytes(); niobuffer.put(tmp, 0, tmp.length); position += charCount; } niobuffer.put(position - 1, (byte)(niobuffer.get(position - 1) | 0x80)); } else { writeUtf8Length(charCount + 1); int charIndex = 0; if (capacity - position >= charCount) { // Try to write 8 bit chars. int position = this.position; for (; charIndex < charCount; charIndex++) { int c = value.charAt(charIndex); if (c > 127) break; niobuffer.put(position++, (byte)c); } this.position = position; niobuffer.position(position); } if (charIndex < charCount) writeString_slow(value, charCount, charIndex); niobuffer.position(position); } } /** Writes the length and CharSequence as UTF8, or null. The string can be read using {@link Input#readString()} or * {@link Input#readStringBuilder()}. * @param value May be null. */ public void writeString (CharSequence value) throws KryoException { if (value == null) { writeByte(0x80); // 0 means null, bit 8 means UTF8. return; } int charCount = value.length(); if (charCount == 0) { writeByte(1 | 0x80); // 1 means empty string, bit 8 means UTF8. return; } writeUtf8Length(charCount + 1); int charIndex = 0; if (capacity - position >= charCount) { // Try to write 8 bit chars. int position = this.position; for (; charIndex < charCount; charIndex++) { int c = value.charAt(charIndex); if (c > 127) break; niobuffer.put(position++, (byte)c); } this.position = position; niobuffer.position(position); } if (charIndex < charCount) writeString_slow(value, charCount, charIndex); niobuffer.position(position); } /** Writes a string that is known to contain only ASCII characters. Non-ASCII strings passed to this method will be corrupted. * Each byte is a 7 bit character with the remaining byte denoting if another character is available. This is slightly more * efficient than {@link #writeString(String)}. The string can be read using {@link Input#readString()} or * {@link Input#readStringBuilder()}. * @param value May be null. */ public void writeAscii (String value) throws KryoException { if (value == null) { writeByte(0x80); // 0 means null, bit 8 means UTF8. return; } int charCount = value.length(); if (charCount == 0) { writeByte(1 | 0x80); // 1 means empty string, bit 8 means UTF8. return; } if (capacity - position < charCount) writeAscii_slow(value, charCount); else { byte[] tmp = value.getBytes(); niobuffer.put(tmp, 0, tmp.length); position += charCount; } niobuffer.put(position - 1, (byte)(niobuffer.get(position - 1) | 0x80)); // Bit 8 means end of ASCII. } /** Writes the length of a string, which is a variable length encoded int except the first byte uses bit 8 to denote UTF8 and * bit 7 to denote if another byte is present. */ private void writeUtf8Length (int value) { if (value >>> 6 == 0) { require(1); niobuffer.put((byte)(value | 0x80)); // Set bit 8. position += 1; } else if (value >>> 13 == 0) { require(2); niobuffer.put((byte)(value | 0x40 | 0x80)); // Set bit 7 and 8. niobuffer.put((byte)(value >>> 6)); position += 2; } else if (value >>> 20 == 0) { require(3); niobuffer.put((byte)(value | 0x40 | 0x80)); // Set bit 7 and 8. niobuffer.put((byte)((value >>> 6) | 0x80)); // Set bit 8. niobuffer.put((byte)(value >>> 13)); position += 3; } else if (value >>> 27 == 0) { require(4); niobuffer.put((byte)(value | 0x40 | 0x80)); // Set bit 7 and 8. niobuffer.put((byte)((value >>> 6) | 0x80)); // Set bit 8. niobuffer.put((byte)((value >>> 13) | 0x80)); // Set bit 8. niobuffer.put((byte)(value >>> 20)); position += 4; } else { require(5); niobuffer.put((byte)(value | 0x40 | 0x80)); // Set bit 7 and 8. niobuffer.put((byte)((value >>> 6) | 0x80)); // Set bit 8. niobuffer.put((byte)((value >>> 13) | 0x80)); // Set bit 8. niobuffer.put((byte)((value >>> 20) | 0x80)); // Set bit 8. niobuffer.put((byte)(value >>> 27)); position += 5; } } private void writeString_slow (CharSequence value, int charCount, int charIndex) { for (; charIndex < charCount; charIndex++) { if (position == capacity) require(Math.min(capacity, charCount - charIndex)); int c = value.charAt(charIndex); if (c <= 0x007F) { niobuffer.put(position++, (byte)c); } else if (c > 0x07FF) { niobuffer.put(position++, (byte)(0xE0 | c >> 12 & 0x0F)); require(2); niobuffer.put(position++, (byte)(0x80 | c >> 6 & 0x3F)); niobuffer.put(position++, (byte)(0x80 | c & 0x3F)); } else { niobuffer.put(position++, (byte)(0xC0 | c >> 6 & 0x1F)); require(1); niobuffer.put(position++, (byte)(0x80 | c & 0x3F)); } } } private void writeAscii_slow (String value, int charCount) throws KryoException { ByteBuffer buffer = this.niobuffer; int charIndex = 0; int charsToWrite = Math.min(charCount, capacity - position); while (charIndex < charCount) { byte[] tmp = new byte[charCount]; value.getBytes(charIndex, charIndex + charsToWrite, tmp, 0); buffer.put(tmp, 0, charsToWrite); // value.getBytes(charIndex, charIndex + charsToWrite, buffer, position); charIndex += charsToWrite; position += charsToWrite; charsToWrite = Math.min(charCount - charIndex, capacity); if (require(charsToWrite)) buffer = this.niobuffer; } } // float /** Writes a 4 byte float. */ public void writeFloat (float value) throws KryoException { require(4); niobuffer.putFloat(value); position += 4; } /** Writes a 1-5 byte float with reduced precision. * @param optimizePositive If true, small positive numbers will be more efficient (1 byte) and small negative numbers will be * inefficient (5 bytes). */ public int writeFloat (float value, float precision, boolean optimizePositive) throws KryoException { return writeInt((int)(value * precision), optimizePositive); } // short /** Writes a 2 byte short. */ public void writeShort (int value) throws KryoException { require(2); niobuffer.putShort((short)value); position += 2; } // long /** Writes an 8 byte long. */ public void writeLong (long value) throws KryoException { require(8); niobuffer.putLong(value); position += 8; } public int writeLong (long value, boolean optimizePositive) throws KryoException { if (!varIntsEnabled) { writeLong(value); return 8; } else return writeVarLong(value, optimizePositive); } public int writeVarLong (long value, boolean optimizePositive) throws KryoException { if (!optimizePositive) value = (value << 1) ^ (value >> 63); int varInt = 0; varInt = (int)(value & 0x7F); value >>>= 7; if (value == 0) { writeByte(varInt); return 1; } varInt |= 0x80; varInt |= ((value & 0x7F) << 8); value >>>= 7; if (value == 0) { niobuffer.order(ByteOrder.LITTLE_ENDIAN); writeInt(varInt); niobuffer.order(byteOrder); position -= 2; niobuffer.position(position); return 2; } varInt |= (0x80 << 8); varInt |= ((value & 0x7F) << 16); value >>>= 7; if (value == 0) { niobuffer.order(ByteOrder.LITTLE_ENDIAN); writeInt(varInt); niobuffer.order(byteOrder); position -= 1; niobuffer.position(position); return 3; } varInt |= (0x80 << 16); varInt |= ((value & 0x7F) << 24); value >>>= 7; if (value == 0) { niobuffer.order(ByteOrder.LITTLE_ENDIAN); writeInt(varInt); niobuffer.order(byteOrder); position -= 0; return 4; } varInt |= (0x80 << 24); long varLong = (varInt & 0xFFFFFFFFL); varLong |= (((long)(value & 0x7F)) << 32); value >>>= 7; if (value == 0) { niobuffer.order(ByteOrder.LITTLE_ENDIAN); writeLong(varLong); niobuffer.order(byteOrder); position -= 3; niobuffer.position(position); return 5; } varLong |= (0x80L << 32); varLong |= (((long)(value & 0x7F)) << 40); value >>>= 7; if (value == 0) { niobuffer.order(ByteOrder.LITTLE_ENDIAN); writeLong(varLong); niobuffer.order(byteOrder); position -= 2; niobuffer.position(position); return 6; } varLong |= (0x80L << 40); varLong |= (((long)(value & 0x7F)) << 48); value >>>= 7; if (value == 0) { niobuffer.order(ByteOrder.LITTLE_ENDIAN); writeLong(varLong); niobuffer.order(byteOrder); position -= 1; niobuffer.position(position); return 7; } varLong |= (0x80L << 48); varLong |= (((long)(value & 0x7F)) << 56); value >>>= 7; if (value == 0) { niobuffer.order(ByteOrder.LITTLE_ENDIAN); writeLong(varLong); niobuffer.order(byteOrder); return 8; } varLong |= (0x80L << 56); niobuffer.order(ByteOrder.LITTLE_ENDIAN); writeLong(varLong); niobuffer.order(byteOrder); write((byte)(value)); return 9; } /** Writes a 1-9 byte long. * @param optimizePositive If true, small positive numbers will be more efficient (1 byte) and small negative numbers will be * inefficient (9 bytes). */ public int writeLongS (long value, boolean optimizePositive) throws KryoException { if (!optimizePositive) value = (value << 1) ^ (value >> 63); if (value >>> 7 == 0) { require(1); niobuffer.put((byte)value); position += 1; return 1; } if (value >>> 14 == 0) { require(2); niobuffer.put((byte)((value & 0x7F) | 0x80)); niobuffer.put((byte)(value >>> 7)); position += 2; return 2; } if (value >>> 21 == 0) { require(3); niobuffer.put((byte)((value & 0x7F) | 0x80)); niobuffer.put((byte)(value >>> 7 | 0x80)); niobuffer.put((byte)(value >>> 14)); position += 3; return 3; } if (value >>> 28 == 0) { require(4); niobuffer.put((byte)((value & 0x7F) | 0x80)); niobuffer.put((byte)(value >>> 7 | 0x80)); niobuffer.put((byte)(value >>> 14 | 0x80)); niobuffer.put((byte)(value >>> 21)); position += 4; return 4; } if (value >>> 35 == 0) { require(5); niobuffer.put((byte)((value & 0x7F) | 0x80)); niobuffer.put((byte)(value >>> 7 | 0x80)); niobuffer.put((byte)(value >>> 14 | 0x80)); niobuffer.put((byte)(value >>> 21 | 0x80)); niobuffer.put((byte)(value >>> 28)); position += 5; return 5; } if (value >>> 42 == 0) { require(6); niobuffer.put((byte)((value & 0x7F) | 0x80)); niobuffer.put((byte)(value >>> 7 | 0x80)); niobuffer.put((byte)(value >>> 14 | 0x80)); niobuffer.put((byte)(value >>> 21 | 0x80)); niobuffer.put((byte)(value >>> 28 | 0x80)); niobuffer.put((byte)(value >>> 35)); position += 6; return 6; } if (value >>> 49 == 0) { require(7); niobuffer.put((byte)((value & 0x7F) | 0x80)); niobuffer.put((byte)(value >>> 7 | 0x80)); niobuffer.put((byte)(value >>> 14 | 0x80)); niobuffer.put((byte)(value >>> 21 | 0x80)); niobuffer.put((byte)(value >>> 28 | 0x80)); niobuffer.put((byte)(value >>> 35 | 0x80)); niobuffer.put((byte)(value >>> 42)); position += 7; return 7; } if (value >>> 56 == 0) { require(8); niobuffer.put((byte)((value & 0x7F) | 0x80)); niobuffer.put((byte)(value >>> 7 | 0x80)); niobuffer.put((byte)(value >>> 14 | 0x80)); niobuffer.put((byte)(value >>> 21 | 0x80)); niobuffer.put((byte)(value >>> 28 | 0x80)); niobuffer.put((byte)(value >>> 35 | 0x80)); niobuffer.put((byte)(value >>> 42 | 0x80)); niobuffer.put((byte)(value >>> 49)); position += 8; return 8; } require(9); niobuffer.put((byte)((value & 0x7F) | 0x80)); niobuffer.put((byte)(value >>> 7 | 0x80)); niobuffer.put((byte)(value >>> 14 | 0x80)); niobuffer.put((byte)(value >>> 21 | 0x80)); niobuffer.put((byte)(value >>> 28 | 0x80)); niobuffer.put((byte)(value >>> 35 | 0x80)); niobuffer.put((byte)(value >>> 42 | 0x80)); niobuffer.put((byte)(value >>> 49 | 0x80)); niobuffer.put((byte)(value >>> 56)); position += 9; return 9; } // boolean /** Writes a 1 byte boolean. */ public void writeBoolean (boolean value) throws KryoException { require(1); niobuffer.put((byte)(value ? 1 : 0)); position++; } // char /** Writes a 2 byte char. */ public void writeChar (char value) throws KryoException { require(2); niobuffer.putChar(value); position += 2; } // double /** Writes an 8 byte double. */ public void writeDouble (double value) throws KryoException { require(8); niobuffer.putDouble(value); position += 8; } /** Writes a 1-9 byte double with reduced precision. * @param optimizePositive If true, small positive numbers will be more efficient (1 byte) and small negative numbers will be * inefficient (9 bytes). */ public int writeDouble (double value, double precision, boolean optimizePositive) throws KryoException { return writeLong((long)(value * precision), optimizePositive); } // Methods implementing bulk operations on arrays of primitive types /** Bulk output of an int array. */ public void writeInts (int[] object) throws KryoException { if (capacity - position >= object.length * 4 && isNativeOrder()) { IntBuffer buf = niobuffer.asIntBuffer(); buf.put(object); position += object.length * 4; } else super.writeInts(object); } /** Bulk output of an long array. */ public void writeLongs (long[] object) throws KryoException { if (capacity - position >= object.length * 8 && isNativeOrder()) { LongBuffer buf = niobuffer.asLongBuffer(); buf.put(object); position += object.length * 8; } else super.writeLongs(object); } /** Bulk output of a float array. */ public void writeFloats (float[] object) throws KryoException { if (capacity - position >= object.length * 4 && isNativeOrder()) { FloatBuffer buf = niobuffer.asFloatBuffer(); buf.put(object); position += object.length * 4; } else super.writeFloats(object); } /** Bulk output of a short array. */ public void writeShorts (short[] object) throws KryoException { if (capacity - position >= object.length * 2 && isNativeOrder()) { ShortBuffer buf = niobuffer.asShortBuffer(); buf.put(object); position += object.length * 2; } else super.writeShorts(object); } /** Bulk output of a char array. */ public void writeChars (char[] object) throws KryoException { if (capacity - position >= object.length * 2 && isNativeOrder()) { CharBuffer buf = niobuffer.asCharBuffer(); buf.put(object); position += object.length * 2; } else super.writeChars(object); } /** Bulk output of a double array. */ public void writeDoubles (double[] object) throws KryoException { if (capacity - position >= object.length * 8 && isNativeOrder()) { DoubleBuffer buf = niobuffer.asDoubleBuffer(); buf.put(object); position += object.length * 8; } else super.writeDoubles(object); } private boolean isNativeOrder () { return byteOrder == nativeOrder; } /** Return current setting for variable length encoding of integers * @return current setting for variable length encoding of integers */ public boolean getVarIntsEnabled () { return varIntsEnabled; } /** Controls if a variable length encoding for integer types should be used when serializers suggest it. * * @param varIntsEnabled */ public void setVarIntsEnabled (boolean varIntsEnabled) { this.varIntsEnabled = varIntsEnabled; } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy