From 62bcd2d208a37a59582c05b63adb088d3fcfb4f4 Mon Sep 17 00:00:00 2001 From: Wojtek Gdela Date: Sat, 14 Oct 2023 19:32:35 +0200 Subject: [PATCH 1/4] Avoid allocation in `BigDecimal` serializer, new method for bytes in Input/Output . --- .../kryo/io/ByteBufferInput.java | 10 +++++ .../kryo/io/ByteBufferOutput.java | 8 ++++ src/com/esotericsoftware/kryo/io/Input.java | 12 ++++++ src/com/esotericsoftware/kryo/io/Output.java | 12 +++++- .../kryo/serializers/DefaultSerializers.java | 37 ++++++------------- 5 files changed, 52 insertions(+), 27 deletions(-) diff --git a/src/com/esotericsoftware/kryo/io/ByteBufferInput.java b/src/com/esotericsoftware/kryo/io/ByteBufferInput.java index cec6d4db6..967f6bda5 100644 --- a/src/com/esotericsoftware/kryo/io/ByteBufferInput.java +++ b/src/com/esotericsoftware/kryo/io/ByteBufferInput.java @@ -352,6 +352,16 @@ public void readBytes (byte[] bytes, int offset, int count) throws KryoException } } + public long readBytesAsLong (int count) { + require(count); + position += count; + long bytes = byteBuffer.get(); + for (int i = 1; i < count; i++) { + bytes = (bytes << 8) | (byteBuffer.get() & 0xFF); + } + return bytes; + } + // int: public int readInt () throws KryoException { diff --git a/src/com/esotericsoftware/kryo/io/ByteBufferOutput.java b/src/com/esotericsoftware/kryo/io/ByteBufferOutput.java index 4061f16db..7447350d7 100644 --- a/src/com/esotericsoftware/kryo/io/ByteBufferOutput.java +++ b/src/com/esotericsoftware/kryo/io/ByteBufferOutput.java @@ -276,6 +276,14 @@ public void writeBytes (byte[] bytes, int offset, int count) throws KryoExceptio } } + public void writeBytesFromLong (long bytes, int count) { + require(count); + position += count; + for (int i = count - 1; i >= 0; i--) { + byteBuffer.put((byte) (bytes >> (i << 3))); + } + } + // int: public void writeInt (int value) throws KryoException { diff --git a/src/com/esotericsoftware/kryo/io/Input.java b/src/com/esotericsoftware/kryo/io/Input.java index 78c1d284c..212364531 100644 --- a/src/com/esotericsoftware/kryo/io/Input.java +++ b/src/com/esotericsoftware/kryo/io/Input.java @@ -373,6 +373,18 @@ public void readBytes (byte[] bytes, int offset, int count) throws KryoException } } + /** Reads count bytes and returns them as long, the last byte read will be the lowest byte in the long. */ + public long readBytesAsLong (int count) { + require(count); + int p = position; + position = p + count; + long bytes = buffer[p++]; + for (int i = 1; i < count; i++) { + bytes = (bytes << 8) | (buffer[p++] & 0xFF); + } + return bytes; + } + // int: /** Reads a 4 byte int. */ diff --git a/src/com/esotericsoftware/kryo/io/Output.java b/src/com/esotericsoftware/kryo/io/Output.java index b1d351c69..3f9c743c4 100644 --- a/src/com/esotericsoftware/kryo/io/Output.java +++ b/src/com/esotericsoftware/kryo/io/Output.java @@ -20,7 +20,6 @@ package com.esotericsoftware.kryo.io; import com.esotericsoftware.kryo.KryoException; -import com.esotericsoftware.kryo.io.KryoBufferOverflowException; import com.esotericsoftware.kryo.util.Pool.Poolable; import com.esotericsoftware.kryo.util.Util; @@ -274,6 +273,17 @@ public void writeBytes (byte[] bytes, int offset, int count) throws KryoExceptio } } + /** Writes count bytes from long, the last byte written is the lowest byte from the long. + * Note the number of bytes is not written. */ + public void writeBytesFromLong (long bytes, int count) { + require(count); + int p = position; + position = p + count; + for (int i = count - 1; i >= 0; i--) { + buffer[p++] = (byte) (bytes >> (i << 3)); + } + } + // int: /** Writes a 4 byte int. */ diff --git a/src/com/esotericsoftware/kryo/serializers/DefaultSerializers.java b/src/com/esotericsoftware/kryo/serializers/DefaultSerializers.java index 32e5cffa8..8c9bb019f 100644 --- a/src/com/esotericsoftware/kryo/serializers/DefaultSerializers.java +++ b/src/com/esotericsoftware/kryo/serializers/DefaultSerializers.java @@ -21,6 +21,7 @@ import static com.esotericsoftware.kryo.Kryo.*; import static com.esotericsoftware.kryo.util.Util.*; +import static java.lang.Long.numberOfLeadingZeros; import com.esotericsoftware.kryo.Kryo; import com.esotericsoftware.kryo.KryoException; @@ -282,27 +283,15 @@ public void write (Kryo kryo, Output output, BigDecimal object) { } // compatible with writing unscaled value represented as BigInteger's bytes - private static void writeUnscaledLong(Output output, long unscaledLong) { - if (unscaledLong >>> 7 == 0) { // optimize for tiny values - output.writeVarInt(2, true); - output.writeByte((byte) unscaledLong); - } else { - byte[] bytes = new byte[8]; - int pos = 8; - do { - bytes[--pos] = (byte) (unscaledLong & 0xFF); - unscaledLong >>= 8; - } while (unscaledLong != 0 && unscaledLong != -1); // out of bits - - if (((bytes[pos] ^ unscaledLong) & 0x80) != 0) { - // sign bit didn't fit in previous byte, need to add another byte - bytes[--pos] = (byte) unscaledLong; - } + private static void writeUnscaledLong (Output output, long unscaledLong) { + int insignificantBits = unscaledLong >= 0 + ? numberOfLeadingZeros(unscaledLong) + : numberOfLeadingZeros(~unscaledLong); + int significantBits = (64 - insignificantBits) + 1; // one more bit is for the sign + int length = (significantBits + (8 - 1)) >> 3; // how many bytes are needed (rounded up) - int length = 8 - pos; - output.writeVarInt(length + 1, true); - output.writeBytes(bytes, pos, length); - } + output.writeByte(length + 1); + output.writeBytesFromLong(unscaledLong, length); } public BigDecimal read (Kryo kryo, Input input, Class type) { @@ -313,15 +302,11 @@ public BigDecimal read (Kryo kryo, Input input, Class type if (length == NULL) return null; length--; - byte[] bytes = input.readBytes(length); if (length > 8) { + byte[] bytes = input.readBytes(length); unscaledBig = new BigInteger(bytes); } else { - unscaledLong = bytes[0]; - for (int i = 1; i < bytes.length; i++) { - unscaledLong <<= 8; - unscaledLong |= (bytes[i] & 0xFF); - } + unscaledLong = input.readBytesAsLong(length); } int scale = input.readInt(false); From 029190b8d143fc45ce362b04532326999466242f Mon Sep 17 00:00:00 2001 From: Wojtek Gdela Date: Tue, 17 Oct 2023 14:38:12 +0200 Subject: [PATCH 2/4] Improve names, tests, checks for writing and reading bytes of a long. --- .../kryo/io/ByteBufferInput.java | 3 +- .../kryo/io/ByteBufferOutput.java | 3 +- src/com/esotericsoftware/kryo/io/Input.java | 3 +- src/com/esotericsoftware/kryo/io/Output.java | 3 +- .../kryo/serializers/DefaultSerializers.java | 2176 ++++++++--------- .../kryo/io/InputOutputTest.java | 42 + 6 files changed, 1138 insertions(+), 1092 deletions(-) diff --git a/src/com/esotericsoftware/kryo/io/ByteBufferInput.java b/src/com/esotericsoftware/kryo/io/ByteBufferInput.java index 967f6bda5..95cd81040 100644 --- a/src/com/esotericsoftware/kryo/io/ByteBufferInput.java +++ b/src/com/esotericsoftware/kryo/io/ByteBufferInput.java @@ -352,7 +352,8 @@ public void readBytes (byte[] bytes, int offset, int count) throws KryoException } } - public long readBytesAsLong (int count) { + public long readLong (int count) { + if (count < 0 || count > 8) throw new IllegalArgumentException("count must be >= 0 and <= 8: " + count); require(count); position += count; long bytes = byteBuffer.get(); diff --git a/src/com/esotericsoftware/kryo/io/ByteBufferOutput.java b/src/com/esotericsoftware/kryo/io/ByteBufferOutput.java index 7447350d7..70dd32bfc 100644 --- a/src/com/esotericsoftware/kryo/io/ByteBufferOutput.java +++ b/src/com/esotericsoftware/kryo/io/ByteBufferOutput.java @@ -276,7 +276,8 @@ public void writeBytes (byte[] bytes, int offset, int count) throws KryoExceptio } } - public void writeBytesFromLong (long bytes, int count) { + public void writeLong (long bytes, int count) { + if (count < 0 || count > 8) throw new IllegalArgumentException("count must be >= 0 and <= 8: " + count); require(count); position += count; for (int i = count - 1; i >= 0; i--) { diff --git a/src/com/esotericsoftware/kryo/io/Input.java b/src/com/esotericsoftware/kryo/io/Input.java index 212364531..24178e73a 100644 --- a/src/com/esotericsoftware/kryo/io/Input.java +++ b/src/com/esotericsoftware/kryo/io/Input.java @@ -374,7 +374,8 @@ public void readBytes (byte[] bytes, int offset, int count) throws KryoException } /** Reads count bytes and returns them as long, the last byte read will be the lowest byte in the long. */ - public long readBytesAsLong (int count) { + public long readLong (int count) { + if (count < 0 || count > 8) throw new IllegalArgumentException("count must be >= 0 and <= 8: " + count); require(count); int p = position; position = p + count; diff --git a/src/com/esotericsoftware/kryo/io/Output.java b/src/com/esotericsoftware/kryo/io/Output.java index 3f9c743c4..9cbd1eae5 100644 --- a/src/com/esotericsoftware/kryo/io/Output.java +++ b/src/com/esotericsoftware/kryo/io/Output.java @@ -275,7 +275,8 @@ public void writeBytes (byte[] bytes, int offset, int count) throws KryoExceptio /** Writes count bytes from long, the last byte written is the lowest byte from the long. * Note the number of bytes is not written. */ - public void writeBytesFromLong (long bytes, int count) { + public void writeLong (long bytes, int count) { + if (count < 0 || count > 8) throw new IllegalArgumentException("count must be >= 0 and <= 8: " + count); require(count); int p = position; position = p + count; diff --git a/src/com/esotericsoftware/kryo/serializers/DefaultSerializers.java b/src/com/esotericsoftware/kryo/serializers/DefaultSerializers.java index 8c9bb019f..88b78426a 100644 --- a/src/com/esotericsoftware/kryo/serializers/DefaultSerializers.java +++ b/src/com/esotericsoftware/kryo/serializers/DefaultSerializers.java @@ -1,1088 +1,1088 @@ -/* Copyright (c) 2008-2023, 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.serializers; - -import static com.esotericsoftware.kryo.Kryo.*; -import static com.esotericsoftware.kryo.util.Util.*; -import static java.lang.Long.numberOfLeadingZeros; - -import com.esotericsoftware.kryo.Kryo; -import com.esotericsoftware.kryo.KryoException; -import com.esotericsoftware.kryo.KryoSerializable; -import com.esotericsoftware.kryo.Registration; -import com.esotericsoftware.kryo.Serializer; -import com.esotericsoftware.kryo.io.Input; -import com.esotericsoftware.kryo.io.Output; - -import java.lang.reflect.Constructor; -import java.math.BigDecimal; -import java.math.BigInteger; -import java.net.MalformedURLException; -import java.net.URI; -import java.net.URISyntaxException; -import java.net.URL; -import java.nio.charset.Charset; -import java.sql.Time; -import java.sql.Timestamp; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.BitSet; -import java.util.Calendar; -import java.util.Collection; -import java.util.Collections; -import java.util.Comparator; -import java.util.Currency; -import java.util.Date; -import java.util.EnumSet; -import java.util.GregorianCalendar; -import java.util.List; -import java.util.Locale; -import java.util.Map; -import java.util.Map.Entry; -import java.util.PriorityQueue; -import java.util.Set; -import java.util.TimeZone; -import java.util.TreeMap; -import java.util.TreeSet; -import java.util.UUID; -import java.util.concurrent.ConcurrentSkipListMap; -import java.util.concurrent.atomic.AtomicBoolean; -import java.util.concurrent.atomic.AtomicInteger; -import java.util.concurrent.atomic.AtomicLong; -import java.util.concurrent.atomic.AtomicReference; -import java.util.regex.Pattern; - -/** Contains many serializer classes that are provided by {@link Kryo#addDefaultSerializer(Class, Class) default}. - * @author Nathan Sweet */ -public class DefaultSerializers { - public static class VoidSerializer extends ImmutableSerializer { - public void write (Kryo kryo, Output output, Object object) { - } - - public Object read (Kryo kryo, Input input, Class type) { - return null; - } - } - - public static class BooleanSerializer extends ImmutableSerializer { - public void write (Kryo kryo, Output output, Boolean object) { - output.writeBoolean(object); - } - - public Boolean read (Kryo kryo, Input input, Class type) { - return input.readBoolean(); - } - } - - public static class ByteSerializer extends ImmutableSerializer { - public void write (Kryo kryo, Output output, Byte object) { - output.writeByte(object); - } - - public Byte read (Kryo kryo, Input input, Class type) { - return input.readByte(); - } - } - - public static class CharSerializer extends ImmutableSerializer { - public void write (Kryo kryo, Output output, Character object) { - output.writeChar(object); - } - - public Character read (Kryo kryo, Input input, Class type) { - return input.readChar(); - } - } - - public static class ShortSerializer extends ImmutableSerializer { - public void write (Kryo kryo, Output output, Short object) { - output.writeShort(object); - } - - public Short read (Kryo kryo, Input input, Class type) { - return input.readShort(); - } - } - - public static class IntSerializer extends ImmutableSerializer { - public void write (Kryo kryo, Output output, Integer object) { - output.writeInt(object, false); - } - - public Integer read (Kryo kryo, Input input, Class type) { - return input.readInt(false); - } - } - - public static class LongSerializer extends ImmutableSerializer { - public void write (Kryo kryo, Output output, Long object) { - output.writeVarLong(object, false); - } - - public Long read (Kryo kryo, Input input, Class type) { - return input.readVarLong(false); - } - } - - public static class FloatSerializer extends ImmutableSerializer { - public void write (Kryo kryo, Output output, Float object) { - output.writeFloat(object); - } - - public Float read (Kryo kryo, Input input, Class type) { - return input.readFloat(); - } - } - - public static class DoubleSerializer extends ImmutableSerializer { - public void write (Kryo kryo, Output output, Double object) { - output.writeDouble(object); - } - - public Double read (Kryo kryo, Input input, Class type) { - return input.readDouble(); - } - } - - /** @see Output#writeString(String) */ - public static class StringSerializer extends ImmutableSerializer { - { - setAcceptsNull(true); - } - - public void write (Kryo kryo, Output output, String object) { - output.writeString(object); - } - - public String read (Kryo kryo, Input input, Class type) { - return input.readString(); - } - } - - /** Serializer for {@link BigInteger} and any subclass. - * @author Tumi (enhacements) */ - public static class BigIntegerSerializer extends ImmutableSerializer { - { - setAcceptsNull(true); - } - - public void write (Kryo kryo, Output output, BigInteger object) { - if (object == null) { - output.writeByte(NULL); - return; - } - // fast-path optimizations for BigInteger.ZERO constant - if (object == BigInteger.ZERO) { - output.writeByte(2); - output.writeByte(0); - return; - } - // default behaviour - byte[] bytes = object.toByteArray(); - output.writeVarInt(bytes.length + 1, true); - output.writeBytes(bytes); - } - - public BigInteger read (Kryo kryo, Input input, Class type) { - int length = input.readVarInt(true); - if (length == NULL) return null; - byte[] bytes = input.readBytes(length - 1); - if (type != BigInteger.class && type != null) { - // Use reflection for subclasses. - return newBigIntegerSubclass(type, bytes); - } - if (length == 2) { - // Fast-path optimizations for BigInteger constants. - switch (bytes[0]) { - case 0: - return BigInteger.ZERO; - case 1: - return BigInteger.ONE; - case 10: - return BigInteger.TEN; - } - } - return new BigInteger(bytes); - } - - private static BigInteger newBigIntegerSubclass(Class type, byte[] bytes) { - try { - Constructor constructor = type.getConstructor(byte[].class); - if (!constructor.isAccessible()) { - try { - constructor.setAccessible(true); - } catch (SecurityException ignored) { - } - } - return constructor.newInstance(bytes); - } catch (Exception ex) { - throw new KryoException(ex); - } - } - } - - /** Serializer for {@link BigDecimal} and any subclass. - * @author Tumi (enhacements) */ - public static class BigDecimalSerializer extends ImmutableSerializer { - { - setAcceptsNull(true); - } - - public void write (Kryo kryo, Output output, BigDecimal object) { - if (object == null) { - output.writeByte(NULL); - return; - } - if (object == BigDecimal.ZERO) { - output.writeVarInt(2, true); - output.writeByte((byte) 0); - output.writeInt(0, false); - return; - } - if (object == BigDecimal.ONE) { - output.writeVarInt(2, true); - output.writeByte((byte) 1); - output.writeInt(0, false); - return; - } - - BigInteger unscaledBig = null; // avoid getting it from BigDecimal, as non-inflated BigDecimal will have to create it - boolean compactForm = object.precision() < 19; // less than nineteen decimal digits for sure fits in a long - if (!compactForm) { - unscaledBig = object.unscaledValue(); // get and remember for possible use in non-compact form - compactForm = unscaledBig.bitLength() <= 63; // check exactly if unscaled value will fit in a long - } - - if (!compactForm) { - byte[] bytes = unscaledBig.toByteArray(); - output.writeVarInt(bytes.length + 1, true); - output.writeBytes(bytes); - } else { - long unscaledLong = object.scaleByPowerOfTen(object.scale()).longValue(); // best way to get unscaled long value without creating unscaled BigInteger on the way - writeUnscaledLong(output, unscaledLong); - } - - output.writeInt(object.scale(), false); - } - - // compatible with writing unscaled value represented as BigInteger's bytes - private static void writeUnscaledLong (Output output, long unscaledLong) { - int insignificantBits = unscaledLong >= 0 - ? numberOfLeadingZeros(unscaledLong) - : numberOfLeadingZeros(~unscaledLong); - int significantBits = (64 - insignificantBits) + 1; // one more bit is for the sign - int length = (significantBits + (8 - 1)) >> 3; // how many bytes are needed (rounded up) - - output.writeByte(length + 1); - output.writeBytesFromLong(unscaledLong, length); - } - - public BigDecimal read (Kryo kryo, Input input, Class type) { - BigInteger unscaledBig = null; - long unscaledLong = 0; - - int length = input.readVarInt(true); - if (length == NULL) return null; - length--; - - if (length > 8) { - byte[] bytes = input.readBytes(length); - unscaledBig = new BigInteger(bytes); - } else { - unscaledLong = input.readBytesAsLong(length); - } - - int scale = input.readInt(false); - if (type != BigDecimal.class && type != null) { - // For subclasses, use reflection - return newBigDecimalSubclass(type, unscaledBig != null ? unscaledBig : BigInteger.valueOf(unscaledLong), scale); - } else { - // For BigDecimal, if possible use factory methods to avoid instantiating BigInteger - if (unscaledBig != null) { - return new BigDecimal(unscaledBig, scale); - } else { - if (scale == 0) { - if (unscaledLong == 0) return BigDecimal.ZERO; - if (unscaledLong == 1) return BigDecimal.ONE; - } - return BigDecimal.valueOf(unscaledLong, scale); - } - } - } - - private static BigDecimal newBigDecimalSubclass(Class type, BigInteger unscaledValue, int scale) { - try { - Constructor constructor = type.getConstructor(BigInteger.class, int.class); - if (!constructor.isAccessible()) { - try { - constructor.setAccessible(true); - } catch (SecurityException ignored) { - } - } - return constructor.newInstance(unscaledValue, scale); - } catch (Exception ex) { - throw new KryoException(ex); - } - } - } - - public static class ClassSerializer extends ImmutableSerializer { - { - setAcceptsNull(true); - } - - public void write (Kryo kryo, Output output, Class type) { - kryo.writeClass(output, type); - if (type != null && (type.isPrimitive() || isWrapperClass(type))) output.writeBoolean(type.isPrimitive()); - } - - public Class read (Kryo kryo, Input input, Class ignored) { - Registration registration = kryo.readClass(input); - if (registration == null) return null; - Class type = registration.getType(); - if (!type.isPrimitive() || input.readBoolean()) return type; - return getWrapperClass(type); - } - } - - /** Serializer for {@link Date}, {@link java.sql.Date}, {@link Time}, {@link Timestamp} and any other subclass. - * @author Tumi */ - public static class DateSerializer extends Serializer { - private Date create (Kryo kryo, Class type, long time) throws KryoException { - if (type == Date.class || type == null) { - return new Date(time); - } - if (type == Timestamp.class) { - return new Timestamp(time); - } - if (type == java.sql.Date.class) { - return new java.sql.Date(time); - } - if (type == Time.class) { - return new Time(time); - } - // other cases, reflection - try { - // Try to avoid invoking the no-args constructor - // (which is expected to initialize the instance with the current time) - Constructor constructor = type.getConstructor(long.class); - if (!constructor.isAccessible()) { - try { - constructor.setAccessible(true); - } catch (SecurityException ignored) { - } - } - return constructor.newInstance(time); - } catch (Exception ex) { - // default strategy - Date d = kryo.newInstance(type); - d.setTime(time); - return d; - } - } - - public void write (Kryo kryo, Output output, Date object) { - output.writeVarLong(object.getTime(), true); - } - - public Date read (Kryo kryo, Input input, Class type) { - return create(kryo, type, input.readVarLong(true)); - } - - public Date copy (Kryo kryo, Date original) { - return create(kryo, original.getClass(), original.getTime()); - } - } - - /** Serializer for {@link Timestamp} which preserves the nanoseconds field. */ - public static class TimestampSerializer extends Serializer { - public void write (Kryo kryo, Output output, Timestamp object) { - output.writeVarLong(integralTimeComponent(object), true); - output.writeVarInt(object.getNanos(), true); - } - - public Timestamp read (Kryo kryo, Input input, Class type) { - return create(input.readVarLong(true), input.readVarInt(true)); - } - - public Timestamp copy (Kryo kryo, Timestamp original) { - return create(integralTimeComponent(original), original.getNanos()); - } - - private long integralTimeComponent (Timestamp object) { - return object.getTime() - (object.getNanos() / 1_000_000); - } - - private Timestamp create (long time, int nanos) { - Timestamp t = new Timestamp(time); - t.setNanos(nanos); - return t; - } - } - - public static class EnumSerializer extends ImmutableSerializer { - { - setAcceptsNull(true); - } - - private Object[] enumConstants; - - public EnumSerializer (Class type) { - enumConstants = type.getEnumConstants(); - // We allow the serialization of the (abstract!) Enum.class (instead of an actual "user" enum), - // which also creates an EnumSerializer instance during Kryo.writeClass with the following trace: - // ClassSerializer.write -> Kryo.writeClass -> DefaultClassResolver.writeClass - // -> Kryo.getDefaultSerializer -> ReflectionSerializerFactory.makeSerializer(kryo, EnumSerializer, Enum.class) - // This EnumSerializer instance is expected to be never called for write/read. - if (enumConstants == null && !Enum.class.equals(type)) - throw new IllegalArgumentException("The type must be an enum: " + type); - } - - public void write (Kryo kryo, Output output, Enum object) { - if (object == null) { - output.writeVarInt(NULL, true); - return; - } - output.writeVarInt(object.ordinal() + 1, true); - } - - public Enum read (Kryo kryo, Input input, Class type) { - int ordinal = input.readVarInt(true); - if (ordinal == NULL) return null; - ordinal--; - if (ordinal < 0 || ordinal > enumConstants.length - 1) - throw new KryoException("Invalid ordinal for enum \"" + type.getName() + "\": " + ordinal); - Object constant = enumConstants[ordinal]; - return (Enum)constant; - } - } - - public static class EnumSetSerializer extends Serializer { - public void write (Kryo kryo, Output output, EnumSet object) { - Serializer serializer; - if (object.isEmpty()) { - EnumSet tmp = EnumSet.complementOf(object); - if (tmp.isEmpty()) throw new KryoException("An EnumSet must have a defined Enum to be serialized."); - serializer = kryo.writeClass(output, tmp.iterator().next().getClass()).getSerializer(); - } else { - serializer = kryo.writeClass(output, object.iterator().next().getClass()).getSerializer(); - } - output.writeVarInt(object.size(), true); - for (Object element : object) - serializer.write(kryo, output, element); - } - - public EnumSet read (Kryo kryo, Input input, Class type) { - Registration registration = kryo.readClass(input); - EnumSet object = EnumSet.noneOf(registration.getType()); - Serializer serializer = registration.getSerializer(); - int length = input.readVarInt(true); - for (int i = 0; i < length; i++) - object.add(serializer.read(kryo, input, null)); - return object; - } - - public EnumSet copy (Kryo kryo, EnumSet original) { - return EnumSet.copyOf(original); - } - } - - /** @author Martin Grotzke */ - public static class CurrencySerializer extends ImmutableSerializer { - { - setAcceptsNull(true); - } - - public void write (Kryo kryo, Output output, Currency object) { - output.writeString(object == null ? null : object.getCurrencyCode()); - } - - public Currency read (Kryo kryo, Input input, Class type) { - String currencyCode = input.readString(); - if (currencyCode == null) return null; - return Currency.getInstance(currencyCode); - } - } - - /** @author Martin Grotzke */ - public static class StringBufferSerializer extends Serializer { - { - setAcceptsNull(true); - } - - public void write (Kryo kryo, Output output, StringBuffer object) { - output.writeString(object == null ? null : object.toString()); - } - - public StringBuffer read (Kryo kryo, Input input, Class type) { - String value = input.readString(); - if (value == null) return null; - return new StringBuffer(value); - } - - public StringBuffer copy (Kryo kryo, StringBuffer original) { - return new StringBuffer(original); - } - } - - /** @author Martin Grotzke */ - public static class StringBuilderSerializer extends Serializer { - { - setAcceptsNull(true); - } - - public void write (Kryo kryo, Output output, StringBuilder object) { - output.writeString(object == null ? null : object.toString()); - } - - public StringBuilder read (Kryo kryo, Input input, Class type) { - return input.readStringBuilder(); - } - - public StringBuilder copy (Kryo kryo, StringBuilder original) { - return new StringBuilder(original); - } - } - - public static class KryoSerializableSerializer extends Serializer { - public void write (Kryo kryo, Output output, KryoSerializable object) { - object.write(kryo, output); - } - - public KryoSerializable read (Kryo kryo, Input input, Class type) { - KryoSerializable object = kryo.newInstance(type); - kryo.reference(object); - object.read(kryo, input); - return object; - } - } - - /** Serializer for lists created via {@link Collections#emptyList()} or that were just assigned the - * {@link Collections#EMPTY_LIST}. - * @author Martin Grotzke */ - public static class CollectionsEmptyListSerializer extends ImmutableSerializer { - public void write (Kryo kryo, Output output, Collection object) { - } - - public Collection read (Kryo kryo, Input input, Class type) { - return Collections.EMPTY_LIST; - } - } - - /** Serializer for maps created via {@link Collections#emptyMap()} or that were just assigned the - * {@link Collections#EMPTY_MAP}. - * @author Martin Grotzke */ - public static class CollectionsEmptyMapSerializer extends ImmutableSerializer { - public void write (Kryo kryo, Output output, Map object) { - } - - public Map read (Kryo kryo, Input input, Class type) { - return Collections.EMPTY_MAP; - } - } - - /** Serializer for sets created via {@link Collections#emptySet()} or that were just assigned the - * {@link Collections#EMPTY_SET}. - * @author Martin Grotzke */ - public static class CollectionsEmptySetSerializer extends ImmutableSerializer { - public void write (Kryo kryo, Output output, Set object) { - } - - public Set read (Kryo kryo, Input input, Class type) { - return Collections.EMPTY_SET; - } - } - - /** Serializer for lists created via {@link Collections#singletonList(Object)}. - * @author Martin Grotzke */ - public static class CollectionsSingletonListSerializer extends Serializer { - public void write (Kryo kryo, Output output, List object) { - kryo.writeClassAndObject(output, object.get(0)); - } - - public List read (Kryo kryo, Input input, Class type) { - return Collections.singletonList(kryo.readClassAndObject(input)); - } - - public List copy (Kryo kryo, List original) { - return Collections.singletonList(kryo.copy(original.get(0))); - } - } - - /** Serializer for maps created via {@link Collections#singletonMap(Object, Object)}. - * @author Martin Grotzke */ - public static class CollectionsSingletonMapSerializer extends Serializer { - public void write (Kryo kryo, Output output, Map object) { - Entry entry = (Entry)object.entrySet().iterator().next(); - kryo.writeClassAndObject(output, entry.getKey()); - kryo.writeClassAndObject(output, entry.getValue()); - } - - public Map read (Kryo kryo, Input input, Class type) { - Object key = kryo.readClassAndObject(input); - Object value = kryo.readClassAndObject(input); - return Collections.singletonMap(key, value); - } - - public Map copy (Kryo kryo, Map original) { - Entry entry = (Entry)original.entrySet().iterator().next(); - return Collections.singletonMap(kryo.copy(entry.getKey()), kryo.copy(entry.getValue())); - } - } - - /** Serializer for sets created via {@link Collections#singleton(Object)}. - * @author Martin Grotzke */ - public static class CollectionsSingletonSetSerializer extends Serializer { - public void write (Kryo kryo, Output output, Set object) { - kryo.writeClassAndObject(output, object.iterator().next()); - } - - public Set read (Kryo kryo, Input input, Class type) { - return Collections.singleton(kryo.readClassAndObject(input)); - } - - public Set copy (Kryo kryo, Set original) { - return Collections.singleton(kryo.copy(original.iterator().next())); - } - } - - /** Serializer for {@link TimeZone}. Assumes the timezones are immutable. - * @author Tumi */ - public static class TimeZoneSerializer extends ImmutableSerializer { - public void write (Kryo kryo, Output output, TimeZone object) { - output.writeString(object.getID()); - } - - public TimeZone read (Kryo kryo, Input input, Class type) { - return TimeZone.getTimeZone(input.readString()); - } - } - - /** Serializer for {@link GregorianCalendar}, java.util.JapaneseImperialCalendar, and sun.util.BuddhistCalendar. - * @author Tumi */ - public static class CalendarSerializer extends Serializer { - // The default value of gregorianCutover. - private static final long DEFAULT_GREGORIAN_CUTOVER = -12219292800000L; - - TimeZoneSerializer timeZoneSerializer = new TimeZoneSerializer(); - - public void write (Kryo kryo, Output output, Calendar object) { - timeZoneSerializer.write(kryo, output, object.getTimeZone()); // can't be null - output.writeVarLong(object.getTimeInMillis(), true); - output.writeBoolean(object.isLenient()); - output.writeInt(object.getFirstDayOfWeek(), true); - output.writeInt(object.getMinimalDaysInFirstWeek(), true); - if (object instanceof GregorianCalendar) - output.writeVarLong(((GregorianCalendar)object).getGregorianChange().getTime(), false); - else - output.writeVarLong(DEFAULT_GREGORIAN_CUTOVER, false); - } - - public Calendar read (Kryo kryo, Input input, Class type) { - Calendar result = Calendar.getInstance(timeZoneSerializer.read(kryo, input, TimeZone.class)); - result.setTimeInMillis(input.readVarLong(true)); - result.setLenient(input.readBoolean()); - result.setFirstDayOfWeek(input.readInt(true)); - result.setMinimalDaysInFirstWeek(input.readInt(true)); - long gregorianChange = input.readVarLong(false); - if (gregorianChange != DEFAULT_GREGORIAN_CUTOVER) - if (result instanceof GregorianCalendar) ((GregorianCalendar)result).setGregorianChange(new Date(gregorianChange)); - return result; - } - - public Calendar copy (Kryo kryo, Calendar original) { - return (Calendar)original.clone(); - } - } - - /** Serializer for {@link TreeMap} and any subclass. - * @author Tumi (enhacements) */ - public static class TreeMapSerializer extends MapSerializer { - protected void writeHeader (Kryo kryo, Output output, TreeMap treeSet) { - kryo.writeClassAndObject(output, treeSet.comparator()); - } - - protected TreeMap create (Kryo kryo, Input input, Class type, int size) { - return createTreeMap(type, (Comparator)kryo.readClassAndObject(input)); - } - - protected TreeMap createCopy (Kryo kryo, TreeMap original) { - return createTreeMap(original.getClass(), original.comparator()); - } - - private TreeMap createTreeMap (Class type, Comparator comparator) { - if (type == TreeMap.class || type == null) return new TreeMap(comparator); - // Use reflection for subclasses. - try { - Constructor constructor = type.getConstructor(Comparator.class); - if (!constructor.isAccessible()) { - try { - constructor.setAccessible(true); - } catch (SecurityException ignored) { - } - } - return (TreeMap)constructor.newInstance(comparator); - } catch (Exception ex) { - throw new KryoException(ex); - } - } - } - - /** Serializer for {@link ConcurrentSkipListMap} and any subclass. - * @author Mr14huashao (enhacements) */ - public static class ConcurrentSkipListMapSerializer extends MapSerializer { - @Override - protected void writeHeader (Kryo kryo, Output output, ConcurrentSkipListMap concurrentSkipListMap) { - kryo.writeClassAndObject(output, concurrentSkipListMap.comparator()); - } - - @Override - protected ConcurrentSkipListMap create (Kryo kryo, Input input, Class type, - int size) { - return createConcurrentSkipListMap(type, (Comparator)kryo.readClassAndObject(input)); - } - - @Override - protected ConcurrentSkipListMap createCopy (Kryo kryo, ConcurrentSkipListMap original) { - return createConcurrentSkipListMap(original.getClass(), original.comparator()); - } - - private ConcurrentSkipListMap createConcurrentSkipListMap (Class type, - Comparator comparator) { - if (type == ConcurrentSkipListMap.class || type == null) { - return new ConcurrentSkipListMap(comparator); - } - // Use reflection for subclasses. - try { - Constructor constructor = type.getConstructor(Comparator.class); - if (!constructor.isAccessible()) { - try { - constructor.setAccessible(true); - } catch (SecurityException ignored) { - } - } - return (ConcurrentSkipListMap)constructor.newInstance(comparator); - } catch (Exception ex) { - throw new KryoException(ex); - } - } - } - - /** Serializer for {@link TreeMap} and any subclass. - * @author Tumi (enhacements) */ - public static class TreeSetSerializer extends CollectionSerializer { - protected void writeHeader (Kryo kryo, Output output, TreeSet treeSet) { - kryo.writeClassAndObject(output, treeSet.comparator()); - } - - protected TreeSet create (Kryo kryo, Input input, Class type, int size) { - return createTreeSet(type, (Comparator)kryo.readClassAndObject(input)); - } - - protected TreeSet createCopy (Kryo kryo, TreeSet original) { - return createTreeSet(original.getClass(), original.comparator()); - } - - private TreeSet createTreeSet (Class type, Comparator comparator) { - if (type == TreeSet.class || type == null) return new TreeSet(comparator); - // Use reflection for subclasses. - try { - Constructor constructor = type.getConstructor(Comparator.class); - if (!constructor.isAccessible()) { - try { - constructor.setAccessible(true); - } catch (SecurityException ignored) { - } - } - return (TreeSet)constructor.newInstance(comparator); - } catch (Exception ex) { - throw new KryoException(ex); - } - } - } - - /** Serializer for {@link PriorityQueue} and any subclass. - * @author Nathan Sweet */ - public static class PriorityQueueSerializer extends CollectionSerializer { - protected void writeHeader (Kryo kryo, Output output, PriorityQueue queue) { - kryo.writeClassAndObject(output, queue.comparator()); - } - - protected PriorityQueue create (Kryo kryo, Input input, Class type, int size) { - return createPriorityQueue(type, size, (Comparator)kryo.readClassAndObject(input)); - } - - protected PriorityQueue createCopy (Kryo kryo, PriorityQueue original) { - return createPriorityQueue(original.getClass(), original.size(), original.comparator()); - } - - private PriorityQueue createPriorityQueue (Class type, int size, Comparator comparator) { - final int initialCapacity = Math.max(size, 1); - if (type == PriorityQueue.class || type == null) return new PriorityQueue(initialCapacity, comparator); - // Use reflection for subclasses. - try { - Constructor constructor = type.getConstructor(int.class, Comparator.class); - if (!constructor.isAccessible()) { - try { - constructor.setAccessible(true); - } catch (SecurityException ignored) { - } - } - return (PriorityQueue)constructor.newInstance(initialCapacity, comparator); - } catch (Exception ex) { - throw new KryoException(ex); - } - } - } - - /** Serializer for {@link Locale} (immutables). - * @author Tumi */ - public static class LocaleSerializer extends ImmutableSerializer { - // Missing constants in j.u.Locale for common locale - public static final Locale SPANISH = new Locale("es", "", ""); - public static final Locale SPAIN = new Locale("es", "ES", ""); - - protected Locale create (String language, String country, String variant) { - // Fast-path for default locale in this system (may not be in the Locale constants list) - Locale defaultLocale = Locale.getDefault(); - if (isSameLocale(defaultLocale, language, country, variant)) return defaultLocale; - // Fast-paths for constants declared in java.util.Locale : - // 1. "US" locale (typical forced default in many applications) - if (defaultLocale != Locale.US && isSameLocale(Locale.US, language, country, variant)) return Locale.US; - // 2. Language-only constant locales - if (isSameLocale(Locale.ENGLISH, language, country, variant)) return Locale.ENGLISH; - if (isSameLocale(Locale.GERMAN, language, country, variant)) return Locale.GERMAN; - if (isSameLocale(SPANISH, language, country, variant)) return SPANISH; - if (isSameLocale(Locale.FRENCH, language, country, variant)) return Locale.FRENCH; - if (isSameLocale(Locale.ITALIAN, language, country, variant)) return Locale.ITALIAN; - if (isSameLocale(Locale.JAPANESE, language, country, variant)) return Locale.JAPANESE; - if (isSameLocale(Locale.KOREAN, language, country, variant)) return Locale.KOREAN; - if (isSameLocale(Locale.SIMPLIFIED_CHINESE, language, country, variant)) return Locale.SIMPLIFIED_CHINESE; - if (isSameLocale(Locale.CHINESE, language, country, variant)) return Locale.CHINESE; - if (isSameLocale(Locale.TRADITIONAL_CHINESE, language, country, variant)) return Locale.TRADITIONAL_CHINESE; - // 2. Language with Country constant locales - if (isSameLocale(Locale.UK, language, country, variant)) return Locale.UK; - if (isSameLocale(Locale.GERMANY, language, country, variant)) return Locale.GERMANY; - if (isSameLocale(SPAIN, language, country, variant)) return SPAIN; - if (isSameLocale(Locale.FRANCE, language, country, variant)) return Locale.FRANCE; - if (isSameLocale(Locale.ITALY, language, country, variant)) return Locale.ITALY; - if (isSameLocale(Locale.JAPAN, language, country, variant)) return Locale.JAPAN; - if (isSameLocale(Locale.KOREA, language, country, variant)) return Locale.KOREA; - // if (isSameLocale(Locale.CHINA, language, country, variant)) // CHINA==SIMPLIFIED_CHINESE, see Locale.java - // return Locale.CHINA; - // if (isSameLocale(Locale.PRC, language, country, variant)) // PRC==SIMPLIFIED_CHINESE, see Locale.java - // return Locale.PRC; - // if (isSameLocale(Locale.TAIWAN, language, country, variant)) // TAIWAN==SIMPLIFIED_CHINESE, see Locale.java - // return Locale.TAIWAN; - if (isSameLocale(Locale.CANADA, language, country, variant)) return Locale.CANADA; - if (isSameLocale(Locale.CANADA_FRENCH, language, country, variant)) return Locale.CANADA_FRENCH; - - return new Locale(language, country, variant); - } - - public void write (Kryo kryo, Output output, Locale l) { - output.writeAscii(l.getLanguage()); - output.writeAscii(l.getCountry()); - output.writeString(l.getVariant()); - } - - public Locale read (Kryo kryo, Input input, Class type) { - String language = input.readString(); - String country = input.readString(); - String variant = input.readString(); - return create(language, country, variant); - } - - protected static boolean isSameLocale (Locale locale, String language, String country, String variant) { - return (locale.getLanguage().equals(language) && locale.getCountry().equals(country) - && locale.getVariant().equals(variant)); - } - } - - /** Serializer for {@link Charset}. */ - public static class CharsetSerializer extends ImmutableSerializer { - public void write (Kryo kryo, Output output, Charset object) { - output.writeString(object.name()); - } - - public Charset read (Kryo kryo, Input input, Class type) { - return Charset.forName(input.readString()); - } - } - - /** Serializer for {@link URL}. */ - public static class URLSerializer extends ImmutableSerializer { - public void write (Kryo kryo, Output output, URL object) { - output.writeString(object.toExternalForm()); - } - - public URL read (Kryo kryo, Input input, Class type) { - try { - return new java.net.URL(input.readString()); - } catch (MalformedURLException ex) { - throw new KryoException(ex); - } - } - } - - /** Serializer for {@link Arrays#asList(Object...)}. */ - public static class ArraysAsListSerializer extends CollectionSerializer { - protected List create (Kryo kryo, Input input, Class type, int size) { - return new ArrayList(size); - } - - public List read (Kryo kryo, Input input, Class type) { - List list = super.read(kryo, input, type); - if (list == null) return null; - return Arrays.asList(list.toArray()); - } - - public List copy (Kryo kryo, List original) { - Object[] copyArr = new Object[original.size()]; - List copy = Arrays.asList(copyArr); - kryo.reference(copy); - for (int i = 0; i < original.size(); i++) { - copyArr[i] = kryo.copy(original.get(i)); - } - return copy; - } - } - - /** Serializer for {@link BitSet} */ - public static class BitSetSerializer extends Serializer { - public void write (Kryo kryo, Output output, BitSet set) { - long[] values = set.toLongArray(); - output.writeVarInt(values.length, true); - output.writeLongs(values, 0, values.length); - } - - public BitSet read (Kryo kryo, Input input, Class type) { - int length = input.readVarInt(true); - long[] values = input.readLongs(length); - return BitSet.valueOf(values); - } - - public BitSet copy (Kryo kryo, BitSet original) { - return BitSet.valueOf(original.toLongArray()); - } - } - - /** Serializer for {@link Pattern} */ - public static class PatternSerializer extends ImmutableSerializer { - public void write (final Kryo kryo, final Output output, final Pattern pattern) { - output.writeString(pattern.pattern()); - output.writeInt(pattern.flags(), true); - } - - public Pattern read (final Kryo kryo, final Input input, final Class patternClass) { - String regex = input.readString(); - int flags = input.readInt(true); - return Pattern.compile(regex, flags); - } - } - - /** Serializer for {@link URI} */ - public static class URISerializer extends ImmutableSerializer { - public void write (Kryo kryo, Output output, URI uri) { - output.writeString(uri.toString()); - } - - public URI read (Kryo kryo, Input input, Class uriClass) { - try { - return new URI(input.readString()); - } catch (URISyntaxException ex) { - throw new KryoException(ex); - } - } - } - - /** Serializer for {@link UUID} */ - public static class UUIDSerializer extends ImmutableSerializer { - public void write (Kryo kryo, Output output, UUID uuid) { - output.writeLong(uuid.getMostSignificantBits()); - output.writeLong(uuid.getLeastSignificantBits()); - } - - public UUID read (final Kryo kryo, final Input input, final Class uuidClass) { - return new UUID(input.readLong(), input.readLong()); - } - } - - /** Serializer for {@link AtomicBoolean} */ - public static class AtomicBooleanSerializer extends Serializer { - public void write (Kryo kryo, Output output, AtomicBoolean object) { - output.writeBoolean(object.get()); - } - - public AtomicBoolean read (Kryo kryo, Input input, Class type) { - return new AtomicBoolean(input.readBoolean()); - } - - public AtomicBoolean copy (Kryo kryo, AtomicBoolean original) { - return new AtomicBoolean(original.get()); - } - } - - /** Serializer for {@link AtomicInteger} */ - public static class AtomicIntegerSerializer extends Serializer { - public void write (Kryo kryo, Output output, AtomicInteger object) { - output.writeInt(object.get()); - } - - public AtomicInteger read (Kryo kryo, Input input, Class type) { - return new AtomicInteger(input.readInt()); - } - - public AtomicInteger copy (Kryo kryo, AtomicInteger original) { - return new AtomicInteger(original.get()); - } - } - - /** Serializer for {@link AtomicLong} */ - public static class AtomicLongSerializer extends Serializer { - public void write (Kryo kryo, Output output, AtomicLong object) { - output.writeLong(object.get()); - } - - public AtomicLong read (Kryo kryo, Input input, Class type) { - return new AtomicLong(input.readLong()); - } - - public AtomicLong copy (Kryo kryo, AtomicLong original) { - return new AtomicLong(original.get()); - } - } - - /** Serializer for {@link AtomicReference} */ - public static class AtomicReferenceSerializer extends Serializer { - public void write (Kryo kryo, Output output, AtomicReference object) { - kryo.writeClassAndObject(output, object.get()); - } - - public AtomicReference read (Kryo kryo, Input input, Class type) { - final Object value = kryo.readClassAndObject(input); - return new AtomicReference(value); - } - - public AtomicReference copy (Kryo kryo, AtomicReference original) { - return new AtomicReference<>(kryo.copy(original.get())); - } - } -} +/* Copyright (c) 2008-2023, 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.serializers; + +import static com.esotericsoftware.kryo.Kryo.*; +import static com.esotericsoftware.kryo.util.Util.*; +import static java.lang.Long.numberOfLeadingZeros; + +import com.esotericsoftware.kryo.Kryo; +import com.esotericsoftware.kryo.KryoException; +import com.esotericsoftware.kryo.KryoSerializable; +import com.esotericsoftware.kryo.Registration; +import com.esotericsoftware.kryo.Serializer; +import com.esotericsoftware.kryo.io.Input; +import com.esotericsoftware.kryo.io.Output; + +import java.lang.reflect.Constructor; +import java.math.BigDecimal; +import java.math.BigInteger; +import java.net.MalformedURLException; +import java.net.URI; +import java.net.URISyntaxException; +import java.net.URL; +import java.nio.charset.Charset; +import java.sql.Time; +import java.sql.Timestamp; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.BitSet; +import java.util.Calendar; +import java.util.Collection; +import java.util.Collections; +import java.util.Comparator; +import java.util.Currency; +import java.util.Date; +import java.util.EnumSet; +import java.util.GregorianCalendar; +import java.util.List; +import java.util.Locale; +import java.util.Map; +import java.util.Map.Entry; +import java.util.PriorityQueue; +import java.util.Set; +import java.util.TimeZone; +import java.util.TreeMap; +import java.util.TreeSet; +import java.util.UUID; +import java.util.concurrent.ConcurrentSkipListMap; +import java.util.concurrent.atomic.AtomicBoolean; +import java.util.concurrent.atomic.AtomicInteger; +import java.util.concurrent.atomic.AtomicLong; +import java.util.concurrent.atomic.AtomicReference; +import java.util.regex.Pattern; + +/** Contains many serializer classes that are provided by {@link Kryo#addDefaultSerializer(Class, Class) default}. + * @author Nathan Sweet */ +public class DefaultSerializers { + public static class VoidSerializer extends ImmutableSerializer { + public void write (Kryo kryo, Output output, Object object) { + } + + public Object read (Kryo kryo, Input input, Class type) { + return null; + } + } + + public static class BooleanSerializer extends ImmutableSerializer { + public void write (Kryo kryo, Output output, Boolean object) { + output.writeBoolean(object); + } + + public Boolean read (Kryo kryo, Input input, Class type) { + return input.readBoolean(); + } + } + + public static class ByteSerializer extends ImmutableSerializer { + public void write (Kryo kryo, Output output, Byte object) { + output.writeByte(object); + } + + public Byte read (Kryo kryo, Input input, Class type) { + return input.readByte(); + } + } + + public static class CharSerializer extends ImmutableSerializer { + public void write (Kryo kryo, Output output, Character object) { + output.writeChar(object); + } + + public Character read (Kryo kryo, Input input, Class type) { + return input.readChar(); + } + } + + public static class ShortSerializer extends ImmutableSerializer { + public void write (Kryo kryo, Output output, Short object) { + output.writeShort(object); + } + + public Short read (Kryo kryo, Input input, Class type) { + return input.readShort(); + } + } + + public static class IntSerializer extends ImmutableSerializer { + public void write (Kryo kryo, Output output, Integer object) { + output.writeInt(object, false); + } + + public Integer read (Kryo kryo, Input input, Class type) { + return input.readInt(false); + } + } + + public static class LongSerializer extends ImmutableSerializer { + public void write (Kryo kryo, Output output, Long object) { + output.writeVarLong(object, false); + } + + public Long read (Kryo kryo, Input input, Class type) { + return input.readVarLong(false); + } + } + + public static class FloatSerializer extends ImmutableSerializer { + public void write (Kryo kryo, Output output, Float object) { + output.writeFloat(object); + } + + public Float read (Kryo kryo, Input input, Class type) { + return input.readFloat(); + } + } + + public static class DoubleSerializer extends ImmutableSerializer { + public void write (Kryo kryo, Output output, Double object) { + output.writeDouble(object); + } + + public Double read (Kryo kryo, Input input, Class type) { + return input.readDouble(); + } + } + + /** @see Output#writeString(String) */ + public static class StringSerializer extends ImmutableSerializer { + { + setAcceptsNull(true); + } + + public void write (Kryo kryo, Output output, String object) { + output.writeString(object); + } + + public String read (Kryo kryo, Input input, Class type) { + return input.readString(); + } + } + + /** Serializer for {@link BigInteger} and any subclass. + * @author Tumi (enhacements) */ + public static class BigIntegerSerializer extends ImmutableSerializer { + { + setAcceptsNull(true); + } + + public void write (Kryo kryo, Output output, BigInteger object) { + if (object == null) { + output.writeByte(NULL); + return; + } + // fast-path optimizations for BigInteger.ZERO constant + if (object == BigInteger.ZERO) { + output.writeByte(2); + output.writeByte(0); + return; + } + // default behaviour + byte[] bytes = object.toByteArray(); + output.writeVarInt(bytes.length + 1, true); + output.writeBytes(bytes); + } + + public BigInteger read (Kryo kryo, Input input, Class type) { + int length = input.readVarInt(true); + if (length == NULL) return null; + byte[] bytes = input.readBytes(length - 1); + if (type != BigInteger.class && type != null) { + // Use reflection for subclasses. + return newBigIntegerSubclass(type, bytes); + } + if (length == 2) { + // Fast-path optimizations for BigInteger constants. + switch (bytes[0]) { + case 0: + return BigInteger.ZERO; + case 1: + return BigInteger.ONE; + case 10: + return BigInteger.TEN; + } + } + return new BigInteger(bytes); + } + + private static BigInteger newBigIntegerSubclass(Class type, byte[] bytes) { + try { + Constructor constructor = type.getConstructor(byte[].class); + if (!constructor.isAccessible()) { + try { + constructor.setAccessible(true); + } catch (SecurityException ignored) { + } + } + return constructor.newInstance(bytes); + } catch (Exception ex) { + throw new KryoException(ex); + } + } + } + + /** Serializer for {@link BigDecimal} and any subclass. + * @author Tumi (enhacements) */ + public static class BigDecimalSerializer extends ImmutableSerializer { + { + setAcceptsNull(true); + } + + public void write (Kryo kryo, Output output, BigDecimal object) { + if (object == null) { + output.writeByte(NULL); + return; + } + if (object == BigDecimal.ZERO) { + output.writeVarInt(2, true); + output.writeByte((byte) 0); + output.writeInt(0, false); + return; + } + if (object == BigDecimal.ONE) { + output.writeVarInt(2, true); + output.writeByte((byte) 1); + output.writeInt(0, false); + return; + } + + BigInteger unscaledBig = null; // avoid getting it from BigDecimal, as non-inflated BigDecimal will have to create it + boolean compactForm = object.precision() < 19; // less than nineteen decimal digits for sure fits in a long + if (!compactForm) { + unscaledBig = object.unscaledValue(); // get and remember for possible use in non-compact form + compactForm = unscaledBig.bitLength() <= 63; // check exactly if unscaled value will fit in a long + } + + if (!compactForm) { + byte[] bytes = unscaledBig.toByteArray(); + output.writeVarInt(bytes.length + 1, true); + output.writeBytes(bytes); + } else { + long unscaledLong = object.scaleByPowerOfTen(object.scale()).longValue(); // best way to get unscaled long value without creating unscaled BigInteger on the way + writeUnscaledLong(output, unscaledLong); + } + + output.writeInt(object.scale(), false); + } + + // compatible with writing unscaled value represented as BigInteger's bytes + private static void writeUnscaledLong (Output output, long unscaledLong) { + int insignificantBits = unscaledLong >= 0 + ? numberOfLeadingZeros(unscaledLong) + : numberOfLeadingZeros(~unscaledLong); + int significantBits = (64 - insignificantBits) + 1; // one more bit is for the sign + int length = (significantBits + (8 - 1)) >> 3; // how many bytes are needed (rounded up) + + output.writeByte(length + 1); + output.writeLong(unscaledLong, length); + } + + public BigDecimal read (Kryo kryo, Input input, Class type) { + BigInteger unscaledBig = null; + long unscaledLong = 0; + + int length = input.readVarInt(true); + if (length == NULL) return null; + length--; + + if (length > 8) { + byte[] bytes = input.readBytes(length); + unscaledBig = new BigInteger(bytes); + } else { + unscaledLong = input.readLong(length); + } + + int scale = input.readInt(false); + if (type != BigDecimal.class && type != null) { + // For subclasses, use reflection + return newBigDecimalSubclass(type, unscaledBig != null ? unscaledBig : BigInteger.valueOf(unscaledLong), scale); + } else { + // For BigDecimal, if possible use factory methods to avoid instantiating BigInteger + if (unscaledBig != null) { + return new BigDecimal(unscaledBig, scale); + } else { + if (scale == 0) { + if (unscaledLong == 0) return BigDecimal.ZERO; + if (unscaledLong == 1) return BigDecimal.ONE; + } + return BigDecimal.valueOf(unscaledLong, scale); + } + } + } + + private static BigDecimal newBigDecimalSubclass(Class type, BigInteger unscaledValue, int scale) { + try { + Constructor constructor = type.getConstructor(BigInteger.class, int.class); + if (!constructor.isAccessible()) { + try { + constructor.setAccessible(true); + } catch (SecurityException ignored) { + } + } + return constructor.newInstance(unscaledValue, scale); + } catch (Exception ex) { + throw new KryoException(ex); + } + } + } + + public static class ClassSerializer extends ImmutableSerializer { + { + setAcceptsNull(true); + } + + public void write (Kryo kryo, Output output, Class type) { + kryo.writeClass(output, type); + if (type != null && (type.isPrimitive() || isWrapperClass(type))) output.writeBoolean(type.isPrimitive()); + } + + public Class read (Kryo kryo, Input input, Class ignored) { + Registration registration = kryo.readClass(input); + if (registration == null) return null; + Class type = registration.getType(); + if (!type.isPrimitive() || input.readBoolean()) return type; + return getWrapperClass(type); + } + } + + /** Serializer for {@link Date}, {@link java.sql.Date}, {@link Time}, {@link Timestamp} and any other subclass. + * @author Tumi */ + public static class DateSerializer extends Serializer { + private Date create (Kryo kryo, Class type, long time) throws KryoException { + if (type == Date.class || type == null) { + return new Date(time); + } + if (type == Timestamp.class) { + return new Timestamp(time); + } + if (type == java.sql.Date.class) { + return new java.sql.Date(time); + } + if (type == Time.class) { + return new Time(time); + } + // other cases, reflection + try { + // Try to avoid invoking the no-args constructor + // (which is expected to initialize the instance with the current time) + Constructor constructor = type.getConstructor(long.class); + if (!constructor.isAccessible()) { + try { + constructor.setAccessible(true); + } catch (SecurityException ignored) { + } + } + return constructor.newInstance(time); + } catch (Exception ex) { + // default strategy + Date d = kryo.newInstance(type); + d.setTime(time); + return d; + } + } + + public void write (Kryo kryo, Output output, Date object) { + output.writeVarLong(object.getTime(), true); + } + + public Date read (Kryo kryo, Input input, Class type) { + return create(kryo, type, input.readVarLong(true)); + } + + public Date copy (Kryo kryo, Date original) { + return create(kryo, original.getClass(), original.getTime()); + } + } + + /** Serializer for {@link Timestamp} which preserves the nanoseconds field. */ + public static class TimestampSerializer extends Serializer { + public void write (Kryo kryo, Output output, Timestamp object) { + output.writeVarLong(integralTimeComponent(object), true); + output.writeVarInt(object.getNanos(), true); + } + + public Timestamp read (Kryo kryo, Input input, Class type) { + return create(input.readVarLong(true), input.readVarInt(true)); + } + + public Timestamp copy (Kryo kryo, Timestamp original) { + return create(integralTimeComponent(original), original.getNanos()); + } + + private long integralTimeComponent (Timestamp object) { + return object.getTime() - (object.getNanos() / 1_000_000); + } + + private Timestamp create (long time, int nanos) { + Timestamp t = new Timestamp(time); + t.setNanos(nanos); + return t; + } + } + + public static class EnumSerializer extends ImmutableSerializer { + { + setAcceptsNull(true); + } + + private Object[] enumConstants; + + public EnumSerializer (Class type) { + enumConstants = type.getEnumConstants(); + // We allow the serialization of the (abstract!) Enum.class (instead of an actual "user" enum), + // which also creates an EnumSerializer instance during Kryo.writeClass with the following trace: + // ClassSerializer.write -> Kryo.writeClass -> DefaultClassResolver.writeClass + // -> Kryo.getDefaultSerializer -> ReflectionSerializerFactory.makeSerializer(kryo, EnumSerializer, Enum.class) + // This EnumSerializer instance is expected to be never called for write/read. + if (enumConstants == null && !Enum.class.equals(type)) + throw new IllegalArgumentException("The type must be an enum: " + type); + } + + public void write (Kryo kryo, Output output, Enum object) { + if (object == null) { + output.writeVarInt(NULL, true); + return; + } + output.writeVarInt(object.ordinal() + 1, true); + } + + public Enum read (Kryo kryo, Input input, Class type) { + int ordinal = input.readVarInt(true); + if (ordinal == NULL) return null; + ordinal--; + if (ordinal < 0 || ordinal > enumConstants.length - 1) + throw new KryoException("Invalid ordinal for enum \"" + type.getName() + "\": " + ordinal); + Object constant = enumConstants[ordinal]; + return (Enum)constant; + } + } + + public static class EnumSetSerializer extends Serializer { + public void write (Kryo kryo, Output output, EnumSet object) { + Serializer serializer; + if (object.isEmpty()) { + EnumSet tmp = EnumSet.complementOf(object); + if (tmp.isEmpty()) throw new KryoException("An EnumSet must have a defined Enum to be serialized."); + serializer = kryo.writeClass(output, tmp.iterator().next().getClass()).getSerializer(); + } else { + serializer = kryo.writeClass(output, object.iterator().next().getClass()).getSerializer(); + } + output.writeVarInt(object.size(), true); + for (Object element : object) + serializer.write(kryo, output, element); + } + + public EnumSet read (Kryo kryo, Input input, Class type) { + Registration registration = kryo.readClass(input); + EnumSet object = EnumSet.noneOf(registration.getType()); + Serializer serializer = registration.getSerializer(); + int length = input.readVarInt(true); + for (int i = 0; i < length; i++) + object.add(serializer.read(kryo, input, null)); + return object; + } + + public EnumSet copy (Kryo kryo, EnumSet original) { + return EnumSet.copyOf(original); + } + } + + /** @author Martin Grotzke */ + public static class CurrencySerializer extends ImmutableSerializer { + { + setAcceptsNull(true); + } + + public void write (Kryo kryo, Output output, Currency object) { + output.writeString(object == null ? null : object.getCurrencyCode()); + } + + public Currency read (Kryo kryo, Input input, Class type) { + String currencyCode = input.readString(); + if (currencyCode == null) return null; + return Currency.getInstance(currencyCode); + } + } + + /** @author Martin Grotzke */ + public static class StringBufferSerializer extends Serializer { + { + setAcceptsNull(true); + } + + public void write (Kryo kryo, Output output, StringBuffer object) { + output.writeString(object == null ? null : object.toString()); + } + + public StringBuffer read (Kryo kryo, Input input, Class type) { + String value = input.readString(); + if (value == null) return null; + return new StringBuffer(value); + } + + public StringBuffer copy (Kryo kryo, StringBuffer original) { + return new StringBuffer(original); + } + } + + /** @author Martin Grotzke */ + public static class StringBuilderSerializer extends Serializer { + { + setAcceptsNull(true); + } + + public void write (Kryo kryo, Output output, StringBuilder object) { + output.writeString(object == null ? null : object.toString()); + } + + public StringBuilder read (Kryo kryo, Input input, Class type) { + return input.readStringBuilder(); + } + + public StringBuilder copy (Kryo kryo, StringBuilder original) { + return new StringBuilder(original); + } + } + + public static class KryoSerializableSerializer extends Serializer { + public void write (Kryo kryo, Output output, KryoSerializable object) { + object.write(kryo, output); + } + + public KryoSerializable read (Kryo kryo, Input input, Class type) { + KryoSerializable object = kryo.newInstance(type); + kryo.reference(object); + object.read(kryo, input); + return object; + } + } + + /** Serializer for lists created via {@link Collections#emptyList()} or that were just assigned the + * {@link Collections#EMPTY_LIST}. + * @author Martin Grotzke */ + public static class CollectionsEmptyListSerializer extends ImmutableSerializer { + public void write (Kryo kryo, Output output, Collection object) { + } + + public Collection read (Kryo kryo, Input input, Class type) { + return Collections.EMPTY_LIST; + } + } + + /** Serializer for maps created via {@link Collections#emptyMap()} or that were just assigned the + * {@link Collections#EMPTY_MAP}. + * @author Martin Grotzke */ + public static class CollectionsEmptyMapSerializer extends ImmutableSerializer { + public void write (Kryo kryo, Output output, Map object) { + } + + public Map read (Kryo kryo, Input input, Class type) { + return Collections.EMPTY_MAP; + } + } + + /** Serializer for sets created via {@link Collections#emptySet()} or that were just assigned the + * {@link Collections#EMPTY_SET}. + * @author Martin Grotzke */ + public static class CollectionsEmptySetSerializer extends ImmutableSerializer { + public void write (Kryo kryo, Output output, Set object) { + } + + public Set read (Kryo kryo, Input input, Class type) { + return Collections.EMPTY_SET; + } + } + + /** Serializer for lists created via {@link Collections#singletonList(Object)}. + * @author Martin Grotzke */ + public static class CollectionsSingletonListSerializer extends Serializer { + public void write (Kryo kryo, Output output, List object) { + kryo.writeClassAndObject(output, object.get(0)); + } + + public List read (Kryo kryo, Input input, Class type) { + return Collections.singletonList(kryo.readClassAndObject(input)); + } + + public List copy (Kryo kryo, List original) { + return Collections.singletonList(kryo.copy(original.get(0))); + } + } + + /** Serializer for maps created via {@link Collections#singletonMap(Object, Object)}. + * @author Martin Grotzke */ + public static class CollectionsSingletonMapSerializer extends Serializer { + public void write (Kryo kryo, Output output, Map object) { + Entry entry = (Entry)object.entrySet().iterator().next(); + kryo.writeClassAndObject(output, entry.getKey()); + kryo.writeClassAndObject(output, entry.getValue()); + } + + public Map read (Kryo kryo, Input input, Class type) { + Object key = kryo.readClassAndObject(input); + Object value = kryo.readClassAndObject(input); + return Collections.singletonMap(key, value); + } + + public Map copy (Kryo kryo, Map original) { + Entry entry = (Entry)original.entrySet().iterator().next(); + return Collections.singletonMap(kryo.copy(entry.getKey()), kryo.copy(entry.getValue())); + } + } + + /** Serializer for sets created via {@link Collections#singleton(Object)}. + * @author Martin Grotzke */ + public static class CollectionsSingletonSetSerializer extends Serializer { + public void write (Kryo kryo, Output output, Set object) { + kryo.writeClassAndObject(output, object.iterator().next()); + } + + public Set read (Kryo kryo, Input input, Class type) { + return Collections.singleton(kryo.readClassAndObject(input)); + } + + public Set copy (Kryo kryo, Set original) { + return Collections.singleton(kryo.copy(original.iterator().next())); + } + } + + /** Serializer for {@link TimeZone}. Assumes the timezones are immutable. + * @author Tumi */ + public static class TimeZoneSerializer extends ImmutableSerializer { + public void write (Kryo kryo, Output output, TimeZone object) { + output.writeString(object.getID()); + } + + public TimeZone read (Kryo kryo, Input input, Class type) { + return TimeZone.getTimeZone(input.readString()); + } + } + + /** Serializer for {@link GregorianCalendar}, java.util.JapaneseImperialCalendar, and sun.util.BuddhistCalendar. + * @author Tumi */ + public static class CalendarSerializer extends Serializer { + // The default value of gregorianCutover. + private static final long DEFAULT_GREGORIAN_CUTOVER = -12219292800000L; + + TimeZoneSerializer timeZoneSerializer = new TimeZoneSerializer(); + + public void write (Kryo kryo, Output output, Calendar object) { + timeZoneSerializer.write(kryo, output, object.getTimeZone()); // can't be null + output.writeVarLong(object.getTimeInMillis(), true); + output.writeBoolean(object.isLenient()); + output.writeInt(object.getFirstDayOfWeek(), true); + output.writeInt(object.getMinimalDaysInFirstWeek(), true); + if (object instanceof GregorianCalendar) + output.writeVarLong(((GregorianCalendar)object).getGregorianChange().getTime(), false); + else + output.writeVarLong(DEFAULT_GREGORIAN_CUTOVER, false); + } + + public Calendar read (Kryo kryo, Input input, Class type) { + Calendar result = Calendar.getInstance(timeZoneSerializer.read(kryo, input, TimeZone.class)); + result.setTimeInMillis(input.readVarLong(true)); + result.setLenient(input.readBoolean()); + result.setFirstDayOfWeek(input.readInt(true)); + result.setMinimalDaysInFirstWeek(input.readInt(true)); + long gregorianChange = input.readVarLong(false); + if (gregorianChange != DEFAULT_GREGORIAN_CUTOVER) + if (result instanceof GregorianCalendar) ((GregorianCalendar)result).setGregorianChange(new Date(gregorianChange)); + return result; + } + + public Calendar copy (Kryo kryo, Calendar original) { + return (Calendar)original.clone(); + } + } + + /** Serializer for {@link TreeMap} and any subclass. + * @author Tumi (enhacements) */ + public static class TreeMapSerializer extends MapSerializer { + protected void writeHeader (Kryo kryo, Output output, TreeMap treeSet) { + kryo.writeClassAndObject(output, treeSet.comparator()); + } + + protected TreeMap create (Kryo kryo, Input input, Class type, int size) { + return createTreeMap(type, (Comparator)kryo.readClassAndObject(input)); + } + + protected TreeMap createCopy (Kryo kryo, TreeMap original) { + return createTreeMap(original.getClass(), original.comparator()); + } + + private TreeMap createTreeMap (Class type, Comparator comparator) { + if (type == TreeMap.class || type == null) return new TreeMap(comparator); + // Use reflection for subclasses. + try { + Constructor constructor = type.getConstructor(Comparator.class); + if (!constructor.isAccessible()) { + try { + constructor.setAccessible(true); + } catch (SecurityException ignored) { + } + } + return (TreeMap)constructor.newInstance(comparator); + } catch (Exception ex) { + throw new KryoException(ex); + } + } + } + + /** Serializer for {@link ConcurrentSkipListMap} and any subclass. + * @author Mr14huashao (enhacements) */ + public static class ConcurrentSkipListMapSerializer extends MapSerializer { + @Override + protected void writeHeader (Kryo kryo, Output output, ConcurrentSkipListMap concurrentSkipListMap) { + kryo.writeClassAndObject(output, concurrentSkipListMap.comparator()); + } + + @Override + protected ConcurrentSkipListMap create (Kryo kryo, Input input, Class type, + int size) { + return createConcurrentSkipListMap(type, (Comparator)kryo.readClassAndObject(input)); + } + + @Override + protected ConcurrentSkipListMap createCopy (Kryo kryo, ConcurrentSkipListMap original) { + return createConcurrentSkipListMap(original.getClass(), original.comparator()); + } + + private ConcurrentSkipListMap createConcurrentSkipListMap (Class type, + Comparator comparator) { + if (type == ConcurrentSkipListMap.class || type == null) { + return new ConcurrentSkipListMap(comparator); + } + // Use reflection for subclasses. + try { + Constructor constructor = type.getConstructor(Comparator.class); + if (!constructor.isAccessible()) { + try { + constructor.setAccessible(true); + } catch (SecurityException ignored) { + } + } + return (ConcurrentSkipListMap)constructor.newInstance(comparator); + } catch (Exception ex) { + throw new KryoException(ex); + } + } + } + + /** Serializer for {@link TreeMap} and any subclass. + * @author Tumi (enhacements) */ + public static class TreeSetSerializer extends CollectionSerializer { + protected void writeHeader (Kryo kryo, Output output, TreeSet treeSet) { + kryo.writeClassAndObject(output, treeSet.comparator()); + } + + protected TreeSet create (Kryo kryo, Input input, Class type, int size) { + return createTreeSet(type, (Comparator)kryo.readClassAndObject(input)); + } + + protected TreeSet createCopy (Kryo kryo, TreeSet original) { + return createTreeSet(original.getClass(), original.comparator()); + } + + private TreeSet createTreeSet (Class type, Comparator comparator) { + if (type == TreeSet.class || type == null) return new TreeSet(comparator); + // Use reflection for subclasses. + try { + Constructor constructor = type.getConstructor(Comparator.class); + if (!constructor.isAccessible()) { + try { + constructor.setAccessible(true); + } catch (SecurityException ignored) { + } + } + return (TreeSet)constructor.newInstance(comparator); + } catch (Exception ex) { + throw new KryoException(ex); + } + } + } + + /** Serializer for {@link PriorityQueue} and any subclass. + * @author Nathan Sweet */ + public static class PriorityQueueSerializer extends CollectionSerializer { + protected void writeHeader (Kryo kryo, Output output, PriorityQueue queue) { + kryo.writeClassAndObject(output, queue.comparator()); + } + + protected PriorityQueue create (Kryo kryo, Input input, Class type, int size) { + return createPriorityQueue(type, size, (Comparator)kryo.readClassAndObject(input)); + } + + protected PriorityQueue createCopy (Kryo kryo, PriorityQueue original) { + return createPriorityQueue(original.getClass(), original.size(), original.comparator()); + } + + private PriorityQueue createPriorityQueue (Class type, int size, Comparator comparator) { + final int initialCapacity = Math.max(size, 1); + if (type == PriorityQueue.class || type == null) return new PriorityQueue(initialCapacity, comparator); + // Use reflection for subclasses. + try { + Constructor constructor = type.getConstructor(int.class, Comparator.class); + if (!constructor.isAccessible()) { + try { + constructor.setAccessible(true); + } catch (SecurityException ignored) { + } + } + return (PriorityQueue)constructor.newInstance(initialCapacity, comparator); + } catch (Exception ex) { + throw new KryoException(ex); + } + } + } + + /** Serializer for {@link Locale} (immutables). + * @author Tumi */ + public static class LocaleSerializer extends ImmutableSerializer { + // Missing constants in j.u.Locale for common locale + public static final Locale SPANISH = new Locale("es", "", ""); + public static final Locale SPAIN = new Locale("es", "ES", ""); + + protected Locale create (String language, String country, String variant) { + // Fast-path for default locale in this system (may not be in the Locale constants list) + Locale defaultLocale = Locale.getDefault(); + if (isSameLocale(defaultLocale, language, country, variant)) return defaultLocale; + // Fast-paths for constants declared in java.util.Locale : + // 1. "US" locale (typical forced default in many applications) + if (defaultLocale != Locale.US && isSameLocale(Locale.US, language, country, variant)) return Locale.US; + // 2. Language-only constant locales + if (isSameLocale(Locale.ENGLISH, language, country, variant)) return Locale.ENGLISH; + if (isSameLocale(Locale.GERMAN, language, country, variant)) return Locale.GERMAN; + if (isSameLocale(SPANISH, language, country, variant)) return SPANISH; + if (isSameLocale(Locale.FRENCH, language, country, variant)) return Locale.FRENCH; + if (isSameLocale(Locale.ITALIAN, language, country, variant)) return Locale.ITALIAN; + if (isSameLocale(Locale.JAPANESE, language, country, variant)) return Locale.JAPANESE; + if (isSameLocale(Locale.KOREAN, language, country, variant)) return Locale.KOREAN; + if (isSameLocale(Locale.SIMPLIFIED_CHINESE, language, country, variant)) return Locale.SIMPLIFIED_CHINESE; + if (isSameLocale(Locale.CHINESE, language, country, variant)) return Locale.CHINESE; + if (isSameLocale(Locale.TRADITIONAL_CHINESE, language, country, variant)) return Locale.TRADITIONAL_CHINESE; + // 2. Language with Country constant locales + if (isSameLocale(Locale.UK, language, country, variant)) return Locale.UK; + if (isSameLocale(Locale.GERMANY, language, country, variant)) return Locale.GERMANY; + if (isSameLocale(SPAIN, language, country, variant)) return SPAIN; + if (isSameLocale(Locale.FRANCE, language, country, variant)) return Locale.FRANCE; + if (isSameLocale(Locale.ITALY, language, country, variant)) return Locale.ITALY; + if (isSameLocale(Locale.JAPAN, language, country, variant)) return Locale.JAPAN; + if (isSameLocale(Locale.KOREA, language, country, variant)) return Locale.KOREA; + // if (isSameLocale(Locale.CHINA, language, country, variant)) // CHINA==SIMPLIFIED_CHINESE, see Locale.java + // return Locale.CHINA; + // if (isSameLocale(Locale.PRC, language, country, variant)) // PRC==SIMPLIFIED_CHINESE, see Locale.java + // return Locale.PRC; + // if (isSameLocale(Locale.TAIWAN, language, country, variant)) // TAIWAN==SIMPLIFIED_CHINESE, see Locale.java + // return Locale.TAIWAN; + if (isSameLocale(Locale.CANADA, language, country, variant)) return Locale.CANADA; + if (isSameLocale(Locale.CANADA_FRENCH, language, country, variant)) return Locale.CANADA_FRENCH; + + return new Locale(language, country, variant); + } + + public void write (Kryo kryo, Output output, Locale l) { + output.writeAscii(l.getLanguage()); + output.writeAscii(l.getCountry()); + output.writeString(l.getVariant()); + } + + public Locale read (Kryo kryo, Input input, Class type) { + String language = input.readString(); + String country = input.readString(); + String variant = input.readString(); + return create(language, country, variant); + } + + protected static boolean isSameLocale (Locale locale, String language, String country, String variant) { + return (locale.getLanguage().equals(language) && locale.getCountry().equals(country) + && locale.getVariant().equals(variant)); + } + } + + /** Serializer for {@link Charset}. */ + public static class CharsetSerializer extends ImmutableSerializer { + public void write (Kryo kryo, Output output, Charset object) { + output.writeString(object.name()); + } + + public Charset read (Kryo kryo, Input input, Class type) { + return Charset.forName(input.readString()); + } + } + + /** Serializer for {@link URL}. */ + public static class URLSerializer extends ImmutableSerializer { + public void write (Kryo kryo, Output output, URL object) { + output.writeString(object.toExternalForm()); + } + + public URL read (Kryo kryo, Input input, Class type) { + try { + return new java.net.URL(input.readString()); + } catch (MalformedURLException ex) { + throw new KryoException(ex); + } + } + } + + /** Serializer for {@link Arrays#asList(Object...)}. */ + public static class ArraysAsListSerializer extends CollectionSerializer { + protected List create (Kryo kryo, Input input, Class type, int size) { + return new ArrayList(size); + } + + public List read (Kryo kryo, Input input, Class type) { + List list = super.read(kryo, input, type); + if (list == null) return null; + return Arrays.asList(list.toArray()); + } + + public List copy (Kryo kryo, List original) { + Object[] copyArr = new Object[original.size()]; + List copy = Arrays.asList(copyArr); + kryo.reference(copy); + for (int i = 0; i < original.size(); i++) { + copyArr[i] = kryo.copy(original.get(i)); + } + return copy; + } + } + + /** Serializer for {@link BitSet} */ + public static class BitSetSerializer extends Serializer { + public void write (Kryo kryo, Output output, BitSet set) { + long[] values = set.toLongArray(); + output.writeVarInt(values.length, true); + output.writeLongs(values, 0, values.length); + } + + public BitSet read (Kryo kryo, Input input, Class type) { + int length = input.readVarInt(true); + long[] values = input.readLongs(length); + return BitSet.valueOf(values); + } + + public BitSet copy (Kryo kryo, BitSet original) { + return BitSet.valueOf(original.toLongArray()); + } + } + + /** Serializer for {@link Pattern} */ + public static class PatternSerializer extends ImmutableSerializer { + public void write (final Kryo kryo, final Output output, final Pattern pattern) { + output.writeString(pattern.pattern()); + output.writeInt(pattern.flags(), true); + } + + public Pattern read (final Kryo kryo, final Input input, final Class patternClass) { + String regex = input.readString(); + int flags = input.readInt(true); + return Pattern.compile(regex, flags); + } + } + + /** Serializer for {@link URI} */ + public static class URISerializer extends ImmutableSerializer { + public void write (Kryo kryo, Output output, URI uri) { + output.writeString(uri.toString()); + } + + public URI read (Kryo kryo, Input input, Class uriClass) { + try { + return new URI(input.readString()); + } catch (URISyntaxException ex) { + throw new KryoException(ex); + } + } + } + + /** Serializer for {@link UUID} */ + public static class UUIDSerializer extends ImmutableSerializer { + public void write (Kryo kryo, Output output, UUID uuid) { + output.writeLong(uuid.getMostSignificantBits()); + output.writeLong(uuid.getLeastSignificantBits()); + } + + public UUID read (final Kryo kryo, final Input input, final Class uuidClass) { + return new UUID(input.readLong(), input.readLong()); + } + } + + /** Serializer for {@link AtomicBoolean} */ + public static class AtomicBooleanSerializer extends Serializer { + public void write (Kryo kryo, Output output, AtomicBoolean object) { + output.writeBoolean(object.get()); + } + + public AtomicBoolean read (Kryo kryo, Input input, Class type) { + return new AtomicBoolean(input.readBoolean()); + } + + public AtomicBoolean copy (Kryo kryo, AtomicBoolean original) { + return new AtomicBoolean(original.get()); + } + } + + /** Serializer for {@link AtomicInteger} */ + public static class AtomicIntegerSerializer extends Serializer { + public void write (Kryo kryo, Output output, AtomicInteger object) { + output.writeInt(object.get()); + } + + public AtomicInteger read (Kryo kryo, Input input, Class type) { + return new AtomicInteger(input.readInt()); + } + + public AtomicInteger copy (Kryo kryo, AtomicInteger original) { + return new AtomicInteger(original.get()); + } + } + + /** Serializer for {@link AtomicLong} */ + public static class AtomicLongSerializer extends Serializer { + public void write (Kryo kryo, Output output, AtomicLong object) { + output.writeLong(object.get()); + } + + public AtomicLong read (Kryo kryo, Input input, Class type) { + return new AtomicLong(input.readLong()); + } + + public AtomicLong copy (Kryo kryo, AtomicLong original) { + return new AtomicLong(original.get()); + } + } + + /** Serializer for {@link AtomicReference} */ + public static class AtomicReferenceSerializer extends Serializer { + public void write (Kryo kryo, Output output, AtomicReference object) { + kryo.writeClassAndObject(output, object.get()); + } + + public AtomicReference read (Kryo kryo, Input input, Class type) { + final Object value = kryo.readClassAndObject(input); + return new AtomicReference(value); + } + + public AtomicReference copy (Kryo kryo, AtomicReference original) { + return new AtomicReference<>(kryo.copy(original.get())); + } + } +} diff --git a/test/com/esotericsoftware/kryo/io/InputOutputTest.java b/test/com/esotericsoftware/kryo/io/InputOutputTest.java index 5cb29e261..6c5774fcf 100644 --- a/test/com/esotericsoftware/kryo/io/InputOutputTest.java +++ b/test/com/esotericsoftware/kryo/io/InputOutputTest.java @@ -511,6 +511,25 @@ private void runLongTest (Output write) throws IOException { write.writeLong(-268435455); write.writeLong(-134217728); write.writeLong(-268435456); + write.writeLong(0, 1); + write.writeLong(63, 1); + write.writeLong(64, 1); + write.writeLong(127, 1); + write.writeLong(128, 2); + write.writeLong(8192, 2); + write.writeLong(16384, 2); + write.writeLong(2097151, 3); + write.writeLong(1048575, 3); + write.writeLong(134217727, 4); + write.writeLong(268435455, 4); + write.writeLong(134217728, 4); + write.writeLong(268435456, 4); + write.writeLong(-2097151, 3); + write.writeLong(-1048575, 3); + write.writeLong(-134217727, 4); + write.writeLong(-268435455, 4); + write.writeLong(-134217728, 4); + write.writeLong(-268435456, 4); assertEquals(1, write.writeVarLong(0, true)); assertEquals(1, write.writeVarLong(0, false)); assertEquals(1, write.writeVarLong(63, true)); @@ -574,6 +593,25 @@ private void runLongTest (Output write) throws IOException { assertEquals(-268435455, read.readLong()); assertEquals(-134217728, read.readLong()); assertEquals(-268435456, read.readLong()); + assertEquals(0, read.readLong(1)); + assertEquals(63, read.readLong(1)); + assertEquals(64, read.readLong(1)); + assertEquals(127, read.readLong(1)); + assertEquals(128, read.readLong(2)); + assertEquals(8192, read.readLong(2)); + assertEquals(16384, read.readLong(2)); + assertEquals(2097151, read.readLong(3)); + assertEquals(1048575, read.readLong(3)); + assertEquals(134217727, read.readLong(4)); + assertEquals(268435455, read.readLong(4)); + assertEquals(134217728, read.readLong(4)); + assertEquals(268435456, read.readLong(4)); + assertEquals(-2097151, read.readLong(3)); + assertEquals(-1048575, read.readLong(3)); + assertEquals(-134217727, read.readLong(4)); + assertEquals(-268435455, read.readLong(4)); + assertEquals(-134217728, read.readLong(4)); + assertEquals(-268435456, read.readLong(4)); assertEquals(0, read.readVarLong(true)); assertEquals(0, read.readVarLong(false)); assertEquals(63, read.readVarLong(true)); @@ -624,11 +662,15 @@ private void runLongTest (Output write) throws IOException { write.writeLong(value); write.writeVarLong(value, true); write.writeVarLong(value, false); + int numOfBytes = (i % 8) + 1; + write.writeLong(value, numOfBytes); read.setBuffer(write.toBytes()); assertEquals(value, read.readLong()); assertEquals(value, read.readVarLong(true)); assertEquals(value, read.readVarLong(false)); + long numOfBytesMask = numOfBytes == 8 ? -1 : (1L << numOfBytes * 8) - 1; + assertEquals(value & numOfBytesMask, read.readLong(numOfBytes) & numOfBytesMask); } } From 969889934ffdc0657b09e5ded0d9c850bb6a4091 Mon Sep 17 00:00:00 2001 From: Wojtek Gdela Date: Tue, 17 Oct 2023 14:49:42 +0200 Subject: [PATCH 3/4] Add to Input/Output methods for writing and reading bytes of an int. --- .../kryo/io/ByteBufferInput.java | 11 +++++ .../kryo/io/ByteBufferOutput.java | 9 ++++ src/com/esotericsoftware/kryo/io/Input.java | 13 ++++++ src/com/esotericsoftware/kryo/io/Output.java | 12 ++++++ .../kryo/io/InputOutputTest.java | 42 +++++++++++++++++++ 5 files changed, 87 insertions(+) diff --git a/src/com/esotericsoftware/kryo/io/ByteBufferInput.java b/src/com/esotericsoftware/kryo/io/ByteBufferInput.java index 95cd81040..f15e5d437 100644 --- a/src/com/esotericsoftware/kryo/io/ByteBufferInput.java +++ b/src/com/esotericsoftware/kryo/io/ByteBufferInput.java @@ -352,6 +352,17 @@ public void readBytes (byte[] bytes, int offset, int count) throws KryoException } } + public int readInt (int count) { + if (count < 0 || count > 4) throw new IllegalArgumentException("count must be >= 0 and <= 4: " + count); + require(count); + position += count; + int bytes = byteBuffer.get(); + for (int i = 1; i < count; i++) { + bytes = (bytes << 8) | (byteBuffer.get() & 0xFF); + } + return bytes; + } + public long readLong (int count) { if (count < 0 || count > 8) throw new IllegalArgumentException("count must be >= 0 and <= 8: " + count); require(count); diff --git a/src/com/esotericsoftware/kryo/io/ByteBufferOutput.java b/src/com/esotericsoftware/kryo/io/ByteBufferOutput.java index 70dd32bfc..e3f4fb664 100644 --- a/src/com/esotericsoftware/kryo/io/ByteBufferOutput.java +++ b/src/com/esotericsoftware/kryo/io/ByteBufferOutput.java @@ -276,6 +276,15 @@ public void writeBytes (byte[] bytes, int offset, int count) throws KryoExceptio } } + public void writeInt (int bytes, int count) { + if (count < 0 || count > 4) throw new IllegalArgumentException("count must be >= 0 and <= 4: " + count); + require(count); + position += count; + for (int i = count - 1; i >= 0; i--) { + byteBuffer.put((byte) (bytes >> (i << 3))); + } + } + public void writeLong (long bytes, int count) { if (count < 0 || count > 8) throw new IllegalArgumentException("count must be >= 0 and <= 8: " + count); require(count); diff --git a/src/com/esotericsoftware/kryo/io/Input.java b/src/com/esotericsoftware/kryo/io/Input.java index 24178e73a..ec4d6af08 100644 --- a/src/com/esotericsoftware/kryo/io/Input.java +++ b/src/com/esotericsoftware/kryo/io/Input.java @@ -373,6 +373,19 @@ public void readBytes (byte[] bytes, int offset, int count) throws KryoException } } + /** Reads count bytes and returns them as int, the last byte read will be the lowest byte in the int. */ + public int readInt (int count) { + if (count < 0 || count > 4) throw new IllegalArgumentException("count must be >= 0 and <= 4: " + count); + require(count); + int p = position; + position = p + count; + int bytes = buffer[p++]; + for (int i = 1; i < count; i++) { + bytes = (bytes << 8) | (buffer[p++] & 0xFF); + } + return bytes; + } + /** Reads count bytes and returns them as long, the last byte read will be the lowest byte in the long. */ public long readLong (int count) { if (count < 0 || count > 8) throw new IllegalArgumentException("count must be >= 0 and <= 8: " + count); diff --git a/src/com/esotericsoftware/kryo/io/Output.java b/src/com/esotericsoftware/kryo/io/Output.java index 9cbd1eae5..cfec8bb50 100644 --- a/src/com/esotericsoftware/kryo/io/Output.java +++ b/src/com/esotericsoftware/kryo/io/Output.java @@ -273,6 +273,18 @@ public void writeBytes (byte[] bytes, int offset, int count) throws KryoExceptio } } + /** Writes count bytes from long, the last byte written is the lowest byte from the long. + * Note the number of bytes is not written. */ + public void writeInt (int bytes, int count) { + if (count < 0 || count > 4) throw new IllegalArgumentException("count must be >= 0 and <= 4: " + count); + require(count); + int p = position; + position = p + count; + for (int i = count - 1; i >= 0; i--) { + buffer[p++] = (byte) (bytes >> (i << 3)); + } + } + /** Writes count bytes from long, the last byte written is the lowest byte from the long. * Note the number of bytes is not written. */ public void writeLong (long bytes, int count) { diff --git a/test/com/esotericsoftware/kryo/io/InputOutputTest.java b/test/com/esotericsoftware/kryo/io/InputOutputTest.java index 6c5774fcf..5b48be080 100644 --- a/test/com/esotericsoftware/kryo/io/InputOutputTest.java +++ b/test/com/esotericsoftware/kryo/io/InputOutputTest.java @@ -348,6 +348,25 @@ private void runIntTest (Output write) throws IOException { write.writeInt(-268435455); write.writeInt(-134217728); write.writeInt(-268435456); + write.writeInt(0, 1); + write.writeInt(63, 1); + write.writeInt(64, 1); + write.writeInt(127, 1); + write.writeInt(128, 2); + write.writeInt(8192, 2); + write.writeInt(16384, 2); + write.writeInt(2097151, 3); + write.writeInt(1048575, 3); + write.writeInt(134217727, 4); + write.writeInt(268435455, 4); + write.writeInt(134217728, 4); + write.writeInt(268435456, 4); + write.writeInt(-2097151, 3); + write.writeInt(-1048575, 3); + write.writeInt(-134217727, 4); + write.writeInt(-268435455, 4); + write.writeInt(-134217728, 4); + write.writeInt(-268435456, 4); assertEquals(1, write.writeVarInt(0, true)); assertEquals(1, write.writeVarInt(0, false)); assertEquals(1, write.writeVarInt(63, true)); @@ -417,6 +436,25 @@ private void runIntTest (Output write) throws IOException { assertEquals(-268435455, read.readInt()); assertEquals(-134217728, read.readInt()); assertEquals(-268435456, read.readInt()); + assertEquals(0, read.readInt(1)); + assertEquals(63, read.readInt(1)); + assertEquals(64, read.readInt(1)); + assertEquals(127, read.readInt(1)); + assertEquals(128, read.readInt(2)); + assertEquals(8192, read.readInt(2)); + assertEquals(16384, read.readInt(2)); + assertEquals(2097151, read.readInt(3)); + assertEquals(1048575, read.readInt(3)); + assertEquals(134217727, read.readInt(4)); + assertEquals(268435455, read.readInt(4)); + assertEquals(134217728, read.readInt(4)); + assertEquals(268435456, read.readInt(4)); + assertEquals(-2097151, read.readInt(3)); + assertEquals(-1048575, read.readInt(3)); + assertEquals(-134217727, read.readInt(4)); + assertEquals(-268435455, read.readInt(4)); + assertEquals(-134217728, read.readInt(4)); + assertEquals(-268435456, read.readInt(4)); assertTrue(read.canReadVarInt()); assertTrue(read.canReadVarInt()); assertTrue(read.canReadVarInt()); @@ -477,11 +515,15 @@ private void runIntTest (Output write) throws IOException { write.writeInt(value); write.writeVarInt(value, true); write.writeVarInt(value, false); + int numOfBytes = (i % 4) + 1; + write.writeInt(value, numOfBytes); read.setBuffer(write.toBytes()); assertEquals(value, read.readInt()); assertEquals(value, read.readVarInt(true)); assertEquals(value, read.readVarInt(false)); + int numOfBytesMask = numOfBytes == 4 ? -1 : (1 << numOfBytes * 8) - 1; + assertEquals(value & numOfBytesMask, read.readInt(numOfBytes) & numOfBytesMask); } } From 813ae4b778b76c9031f1d1977f35d90304564956 Mon Sep 17 00:00:00 2001 From: Wojtek Gdela Date: Tue, 17 Oct 2023 16:42:44 +0200 Subject: [PATCH 4/4] Optimize writing and reading bytes of a long/int by unrolling loop. --- .../kryo/io/ByteBufferInput.java | 32 +++++++++-------- .../kryo/io/ByteBufferOutput.java | 31 +++++++++++------ src/com/esotericsoftware/kryo/io/Input.java | 34 +++++++++++++------ src/com/esotericsoftware/kryo/io/Output.java | 32 +++++++++++++---- 4 files changed, 85 insertions(+), 44 deletions(-) diff --git a/src/com/esotericsoftware/kryo/io/ByteBufferInput.java b/src/com/esotericsoftware/kryo/io/ByteBufferInput.java index f15e5d437..1db08a005 100644 --- a/src/com/esotericsoftware/kryo/io/ByteBufferInput.java +++ b/src/com/esotericsoftware/kryo/io/ByteBufferInput.java @@ -356,22 +356,24 @@ public int readInt (int count) { if (count < 0 || count > 4) throw new IllegalArgumentException("count must be >= 0 and <= 4: " + count); require(count); position += count; - int bytes = byteBuffer.get(); - for (int i = 1; i < count; i++) { - bytes = (bytes << 8) | (byteBuffer.get() & 0xFF); - } - return bytes; - } - - public long readLong (int count) { - if (count < 0 || count > 8) throw new IllegalArgumentException("count must be >= 0 and <= 8: " + count); - require(count); - position += count; - long bytes = byteBuffer.get(); - for (int i = 1; i < count; i++) { - bytes = (bytes << 8) | (byteBuffer.get() & 0xFF); + ByteBuffer byteBuffer = this.byteBuffer; + switch (count) { + case 1: + return byteBuffer.get(); + case 2: + return byteBuffer.get() << 8 + | byteBuffer.get() & 0xFF; + case 3: + return byteBuffer.get() << 16 + | (byteBuffer.get() & 0xFF) << 8 + | byteBuffer.get() & 0xFF; + case 4: + return byteBuffer.get() << 24 + | (byteBuffer.get() & 0xFF) << 16 + | (byteBuffer.get() & 0xFF) << 8 + | byteBuffer.get() & 0xFF; } - return bytes; + throw new IllegalStateException(); // impossible } // int: diff --git a/src/com/esotericsoftware/kryo/io/ByteBufferOutput.java b/src/com/esotericsoftware/kryo/io/ByteBufferOutput.java index e3f4fb664..07bdc9d30 100644 --- a/src/com/esotericsoftware/kryo/io/ByteBufferOutput.java +++ b/src/com/esotericsoftware/kryo/io/ByteBufferOutput.java @@ -280,17 +280,26 @@ public void writeInt (int bytes, int count) { if (count < 0 || count > 4) throw new IllegalArgumentException("count must be >= 0 and <= 4: " + count); require(count); position += count; - for (int i = count - 1; i >= 0; i--) { - byteBuffer.put((byte) (bytes >> (i << 3))); - } - } - - public void writeLong (long bytes, int count) { - if (count < 0 || count > 8) throw new IllegalArgumentException("count must be >= 0 and <= 8: " + count); - require(count); - position += count; - for (int i = count - 1; i >= 0; i--) { - byteBuffer.put((byte) (bytes >> (i << 3))); + ByteBuffer byteBuffer = this.byteBuffer; + switch (count) { + case 1: + byteBuffer.put((byte)bytes); + break; + case 2: + byteBuffer.put((byte)(bytes >> 8)); + byteBuffer.put((byte)bytes); + break; + case 3: + byteBuffer.put((byte)(bytes >> 16)); + byteBuffer.put((byte)(bytes >> 8)); + byteBuffer.put((byte)bytes); + break; + case 4: + byteBuffer.put((byte)(bytes >> 24)); + byteBuffer.put((byte)(bytes >> 16)); + byteBuffer.put((byte)(bytes >> 8)); + byteBuffer.put((byte)bytes); + break; } } diff --git a/src/com/esotericsoftware/kryo/io/Input.java b/src/com/esotericsoftware/kryo/io/Input.java index ec4d6af08..dd72768fe 100644 --- a/src/com/esotericsoftware/kryo/io/Input.java +++ b/src/com/esotericsoftware/kryo/io/Input.java @@ -379,24 +379,36 @@ public int readInt (int count) { require(count); int p = position; position = p + count; - int bytes = buffer[p++]; - for (int i = 1; i < count; i++) { - bytes = (bytes << 8) | (buffer[p++] & 0xFF); + switch (count) { + case 1: + return buffer[p]; + case 2: + return buffer[p] << 8 + | buffer[p+1] & 0xFF; + case 3: + return buffer[p] << 16 + | (buffer[p+1] & 0xFF) << 8 + | buffer[p+2] & 0xFF; + case 4: + return buffer[p] << 24 + | (buffer[p+1] & 0xFF) << 16 + | (buffer[p+2] & 0xFF) << 8 + | buffer[p+3] & 0xFF; } - return bytes; + throw new IllegalStateException(); // impossible } /** Reads count bytes and returns them as long, the last byte read will be the lowest byte in the long. */ public long readLong (int count) { if (count < 0 || count > 8) throw new IllegalArgumentException("count must be >= 0 and <= 8: " + count); - require(count); - int p = position; - position = p + count; - long bytes = buffer[p++]; - for (int i = 1; i < count; i++) { - bytes = (bytes << 8) | (buffer[p++] & 0xFF); + if (count <= 4) { + return readInt(count); + } else { + require(count); + long highBytes = ((long) readInt(count - 4)) << 32; + long lowBytes = ((long) readInt(4)) & (1L << 32) - 1; + return highBytes | lowBytes; } - return bytes; } // int: diff --git a/src/com/esotericsoftware/kryo/io/Output.java b/src/com/esotericsoftware/kryo/io/Output.java index cfec8bb50..dc7e6ac4a 100644 --- a/src/com/esotericsoftware/kryo/io/Output.java +++ b/src/com/esotericsoftware/kryo/io/Output.java @@ -280,8 +280,25 @@ public void writeInt (int bytes, int count) { require(count); int p = position; position = p + count; - for (int i = count - 1; i >= 0; i--) { - buffer[p++] = (byte) (bytes >> (i << 3)); + switch (count) { + case 1: + buffer[p] = (byte)bytes; + break; + case 2: + buffer[p] = (byte)(bytes >> 8); + buffer[p+1] = (byte)bytes; + break; + case 3: + buffer[p] = (byte)(bytes >> 16); + buffer[p+1] = (byte)(bytes >> 8); + buffer[p+2] = (byte)bytes; + break; + case 4: + buffer[p] = (byte)(bytes >> 24); + buffer[p+1] = (byte)(bytes >> 16); + buffer[p+2] = (byte)(bytes >> 8); + buffer[p+3] = (byte)bytes; + break; } } @@ -289,11 +306,12 @@ public void writeInt (int bytes, int count) { * Note the number of bytes is not written. */ public void writeLong (long bytes, int count) { if (count < 0 || count > 8) throw new IllegalArgumentException("count must be >= 0 and <= 8: " + count); - require(count); - int p = position; - position = p + count; - for (int i = count - 1; i >= 0; i--) { - buffer[p++] = (byte) (bytes >> (i << 3)); + if (count <= 4) { + writeInt((int) bytes, count); + } else { + require(count); + writeInt((int) (bytes >> 32), count - 4); + writeInt((int) bytes, 4); } }