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

com.liferay.petra.io.Serializer Maven / Gradle / Ivy

The newest version!
/**
 * Copyright (c) 2000-present Liferay, Inc. All rights reserved.
 *
 * This library is free software; you can redistribute it and/or modify it under
 * the terms of the GNU Lesser General Public License as published by the Free
 * Software Foundation; either version 2.1 of the License, or (at your option)
 * any later version.
 *
 * This library is distributed in the hope that it will be useful, but WITHOUT
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
 * FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more
 * details.
 */

package com.liferay.petra.io;

import com.liferay.petra.io.constants.SerializationConstants;
import com.liferay.petra.lang.CentralizedThreadLocal;
import com.liferay.petra.lang.ClassLoaderPool;

import java.io.IOException;
import java.io.ObjectOutputStream;
import java.io.OutputStream;
import java.io.Serializable;

import java.lang.ref.Reference;
import java.lang.ref.SoftReference;

import java.nio.ByteBuffer;

import java.util.Arrays;

/**
 * Serializes data in a ClassLoader-aware manner.
 *
 * 

* The Serializer can perform better than {@link ObjectOutputStream} and {@link * java.io.DataOutputStream}, with respect to encoding primary types, because it * uses a more compact format (containing no BlockHeader) and simpler call stack * involving {@link BigEndianCodec}, as compared to using an OutputStream * wrapper on top of {@link java.io.Bits}. *

* *

* For Strings, the UTF encoding for ObjectOutputStream and DataOutputStream has * a 2^16=64K length limitation, which is often too restrictive. Serializer has * a 2^32=4G String length limitation, which is generally more than enough. For * pure ASCII character Strings, the encoding performance is almost the same, if * not better, than ObjectOutputStream and DataOutputStream. For Strings * containing non-ASCII characters, the Serializer encodes each char to two * bytes rather than performing UTF encoding. There is a trade-off between * CPU/memory performance and compression rate. *

* *

* UTF encoding uses more CPU cycles to detect the unicode range for each char * and the resulting output is variable length, which increases the memory * burden when preparing the decoding buffer. Whereas, encoding each char to two * bytes allows for better CPU/memory performance. Although inefficient with * compression rates in comparison to UTF encoding, the char to two byte * approach significantly simplifies the encoder's logic and the output length * is predictably based on the length of the String, so the decoder can manage * its decoding buffer efficiently. On average, a system uses more ASCII String * scheming than non-ASCII String scheming. In most cases, when all system * internal Strings are ASCII Strings and only Strings holding user input * information can have non-ASCII characters, this Serializer performs best. In * other cases, developers should consider using {@link ObjectOutputStream} or * {@link java.io.DataOutputStream}. *

* *

* For ordinary Objects, all primary type wrappers are encoded to their raw * values with one byte type headers. This is much more efficient than * ObjectOutputStream's serialization format for primary type wrappers. Strings * are output in the same way as {@link #writeString(String)}, but also with one * byte type headers. Objects are serialized by a new ObjectOutputStream, so no * reference handler can be used across Object serialization. This is done * intentionally to isolate each object. The Serializer is highly optimized for * serializing primary types, but is not as good as ObjectOutputStream for * serializing complex objects. *

* *

* On object serialization, the Serializer uses the {@link ClassLoaderPool} to * look up the context name corresponding to the object's ClassLoader. The * context name is written to the serialization stream. On object * deserialization, the {@link Deserializer} uses the ClassLoaderPool to look up * the ClassLoader corresponding to the context name read from the * deserialization stream. ObjectOutputStream and ObjectInputStream lack these * features, making Serializer and Deserializer better choices for * ClassLoader-aware Object serialization/deserialization, especially when * plugins are involved. *

* * @author Shuyang Zhou * @see Deserializer */ public class Serializer { public Serializer() { BufferQueue bufferQueue = _getBufferQueue(); _buffer = bufferQueue.dequeue(); } public ByteBuffer toByteBuffer() { ByteBuffer byteBuffer = ByteBuffer.wrap(Arrays.copyOf(_buffer, _index)); if (_buffer.length <= _THREADLOCAL_BUFFER_SIZE_LIMIT) { BufferQueue bufferQueue = _getBufferQueue(); bufferQueue.enqueue(_buffer); } _buffer = null; return byteBuffer; } public void writeBoolean(boolean b) { BigEndianCodec.putBoolean(_getBuffer(1), _index++, b); } public void writeByte(byte b) { _getBuffer(1)[_index++] = b; } public void writeChar(char c) { BigEndianCodec.putChar(_getBuffer(2), _index, c); _index += 2; } public void writeDouble(double d) { BigEndianCodec.putDouble(_getBuffer(8), _index, d); _index += 8; } public void writeFloat(float f) { BigEndianCodec.putFloat(_getBuffer(4), _index, f); _index += 4; } public void writeInt(int i) { BigEndianCodec.putInt(_getBuffer(4), _index, i); _index += 4; } public void writeLong(long l) { BigEndianCodec.putLong(_getBuffer(8), _index, l); _index += 8; } public void writeObject(Serializable serializable) { // The if block is ordered by frequency for better performance if (serializable == null) { writeByte(SerializationConstants.TC_NULL); return; } else if (serializable instanceof Long) { writeByte(SerializationConstants.TC_LONG); writeLong((Long)serializable); return; } else if (serializable instanceof String) { writeByte(SerializationConstants.TC_STRING); writeString((String)serializable); return; } else if (serializable instanceof Integer) { writeByte(SerializationConstants.TC_INTEGER); writeInt((Integer)serializable); return; } else if (serializable instanceof Boolean) { writeByte(SerializationConstants.TC_BOOLEAN); writeBoolean((Boolean)serializable); return; } else if (serializable instanceof Class) { Class clazz = (Class)serializable; writeByte(SerializationConstants.TC_CLASS); writeString(ClassLoaderPool.getContextName(clazz.getClassLoader())); writeString(clazz.getName()); return; } else if (serializable instanceof Short) { writeByte(SerializationConstants.TC_SHORT); writeShort((Short)serializable); return; } else if (serializable instanceof Character) { writeByte(SerializationConstants.TC_CHARACTER); writeChar((Character)serializable); return; } else if (serializable instanceof Byte) { writeByte(SerializationConstants.TC_BYTE); writeByte((Byte)serializable); return; } else if (serializable instanceof Double) { writeByte(SerializationConstants.TC_DOUBLE); writeDouble((Double)serializable); return; } else if (serializable instanceof Float) { writeByte(SerializationConstants.TC_FLOAT); writeFloat((Float)serializable); return; } else { writeByte(SerializationConstants.TC_OBJECT); } try { ObjectOutputStream objectOutputStream = new AnnotatedObjectOutputStream(new BufferOutputStream()); objectOutputStream.writeObject(serializable); objectOutputStream.flush(); } catch (IOException ioException) { throw new RuntimeException( "Unable to write ordinary serializable object " + serializable, ioException); } } public void writeShort(short s) { BigEndianCodec.putShort(_getBuffer(2), _index, s); _index += 2; } public void writeString(String s) { int length = s.length(); boolean asciiCode = true; for (int i = 0; i < length; i++) { char c = s.charAt(i); if ((c == 0) || (c > 127)) { asciiCode = false; break; } } if (asciiCode) { byte[] buffer = _getBuffer(length + 5); BigEndianCodec.putBoolean(buffer, _index++, asciiCode); BigEndianCodec.putInt(buffer, _index, length); _index += 4; for (int i = 0; i < length; i++) { char c = s.charAt(i); buffer[_index++] = (byte)c; } } else { byte[] buffer = _getBuffer((length * 2) + 5); BigEndianCodec.putBoolean(buffer, _index++, asciiCode); BigEndianCodec.putInt(buffer, _index, length); _index += 4; for (int i = 0; i < length; i++) { char c = s.charAt(i); BigEndianCodec.putChar(buffer, _index, c); _index += 2; } } } public void writeTo(OutputStream outputStream) throws IOException { outputStream.write(_buffer, 0, _index); if (_buffer.length <= _THREADLOCAL_BUFFER_SIZE_LIMIT) { BufferQueue bufferQueue = _getBufferQueue(); bufferQueue.enqueue(_buffer); } _buffer = null; } /** * Returns the required buffer length. This method is final so JIT can * perform an inline expansion. * * @param ensureExtraSpace the extra byte space required to meet the * buffer's minimum length * @return the buffer value */ private byte[] _getBuffer(int ensureExtraSpace) { int minSize = _index + ensureExtraSpace; if (minSize < 0) { // Cannot create byte[] with length longer than Integer.MAX_VALUE throw new OutOfMemoryError(); } int oldSize = _buffer.length; if (minSize > oldSize) { int newSize = oldSize << 1; if (newSize < minSize) { // Overflow and insufficient growth protection newSize = minSize; } _buffer = Arrays.copyOf(_buffer, newSize); } return _buffer; } private BufferQueue _getBufferQueue() { Reference reference = _bufferQueueThreadLocal.get(); BufferQueue bufferQueue = null; if (reference != null) { bufferQueue = reference.get(); } if (bufferQueue == null) { bufferQueue = new BufferQueue(); _bufferQueueThreadLocal.set(new SoftReference<>(bufferQueue)); } return bufferQueue; } private static final int _THREADLOCAL_BUFFER_COUNT_LIMIT; private static final int _THREADLOCAL_BUFFER_COUNT_MIN = 8; private static final int _THREADLOCAL_BUFFER_SIZE_LIMIT; private static final int _THREADLOCAL_BUFFER_SIZE_MIN = 16 * 1024; /** * Softens the local thread's pooled buffer memory. * *

* Technically, we should soften each pooled buffer individually to achieve * the best garbage collection (GC) interaction. However, that increases * complexity of pooled buffer access and also burdens the GC's {@link * SoftReference} process, hurting performance. *

* *

* Here, the entire ThreadLocal BufferQueue is softened. For threads that do * serializing often, its BufferQueue will most likely stay valid. For * threads that do serializing only occasionally, its BufferQueue will most * likely be released by GC. *

*/ private static final ThreadLocal> _bufferQueueThreadLocal = new CentralizedThreadLocal<>(false); static { int threadLocalBufferCountLimit = Integer.getInteger( Serializer.class.getName() + ".thread.local.buffer.count.limit", 0); if (threadLocalBufferCountLimit < _THREADLOCAL_BUFFER_COUNT_MIN) { threadLocalBufferCountLimit = _THREADLOCAL_BUFFER_COUNT_MIN; } _THREADLOCAL_BUFFER_COUNT_LIMIT = threadLocalBufferCountLimit; int threadLocalBufferSizeLimit = Integer.getInteger( Serializer.class.getName() + ".thread.local.buffer.size.limit", 0); if (threadLocalBufferSizeLimit < _THREADLOCAL_BUFFER_SIZE_MIN) { threadLocalBufferSizeLimit = _THREADLOCAL_BUFFER_SIZE_MIN; } _THREADLOCAL_BUFFER_SIZE_LIMIT = threadLocalBufferSizeLimit; } private byte[] _buffer; private int _index; private static class BufferNode { private BufferNode(byte[] buffer) { _buffer = buffer; } private byte[] _buffer; private BufferNode _next; } /** * Represents a descending byte[] queue ordered by array * length. * *

* The queue is small enough to simply use a linear scan search for * maintaining its order. The entire queue data is held by a {@link * SoftReference}, so when necessary, GC can release the whole buffer cache. *

*/ private static class BufferQueue { public byte[] dequeue() { if (_headBufferNode == null) { return new byte[_THREADLOCAL_BUFFER_SIZE_MIN]; } BufferNode bufferNode = _headBufferNode; _headBufferNode = _headBufferNode._next; // Help GC bufferNode._next = null; return bufferNode._buffer; } public void enqueue(byte[] buffer) { BufferNode bufferNode = new BufferNode(buffer); if (_headBufferNode == null) { _headBufferNode = bufferNode; _count = 1; return; } BufferNode previousBufferNode = null; BufferNode currentBufferNode = _headBufferNode; while ((currentBufferNode != null) && (currentBufferNode._buffer.length > bufferNode._buffer.length)) { previousBufferNode = currentBufferNode; currentBufferNode = currentBufferNode._next; } if (previousBufferNode == null) { bufferNode._next = _headBufferNode; _headBufferNode = bufferNode; } else { bufferNode._next = currentBufferNode; previousBufferNode._next = bufferNode; } if (++_count > _THREADLOCAL_BUFFER_COUNT_LIMIT) { if (previousBufferNode == null) { previousBufferNode = _headBufferNode; } currentBufferNode = previousBufferNode._next; while (currentBufferNode._next != null) { previousBufferNode = currentBufferNode; currentBufferNode = currentBufferNode._next; } // Dereference previousBufferNode._next = null; // Help GC currentBufferNode._buffer = null; currentBufferNode._next = null; } } private int _count; private BufferNode _headBufferNode; } private class BufferOutputStream extends OutputStream { @Override public void write(byte[] bytes) { write(bytes, 0, bytes.length); } @Override public void write(byte[] bytes, int offset, int length) { System.arraycopy(bytes, offset, _getBuffer(length), _index, length); _index += length; } @Override public void write(int b) { _getBuffer(1)[_index++] = (byte)b; } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy