com.esotericsoftware.kryo.serializers.ExternalizableSerializer Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of redisson-all Show documentation
Show all versions of redisson-all Show documentation
Easy Redis Java client and Real-Time Data Platform. Valkey compatible. Sync/Async/RxJava3/Reactive API. Client side caching. Over 50 Redis based Java objects and services: JCache API, Apache Tomcat, Hibernate, Spring, Set, Multimap, SortedSet, Map, List, Queue, Deque, Semaphore, Lock, AtomicLong, Map Reduce, Bloom filter, Scheduler, RPC
The newest version!
/* 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 com.esotericsoftware.kryo.Kryo;
import com.esotericsoftware.kryo.KryoException;
import com.esotericsoftware.kryo.Serializer;
import com.esotericsoftware.kryo.io.Input;
import com.esotericsoftware.kryo.io.KryoObjectInput;
import com.esotericsoftware.kryo.io.KryoObjectOutput;
import com.esotericsoftware.kryo.io.Output;
import com.esotericsoftware.kryo.util.ObjectMap;
import java.io.Externalizable;
import java.io.ObjectInput;
import java.io.ObjectOutput;
import java.lang.reflect.Method;
/** Writes using the objects externalizable interface if it can reliably do so. Typically, a object can be efficiently written
* with Kryo and Java's externalizable interface. However, there may be behavior problems if the class uses either the
* 'readResolve' or 'writeReplace' methods. We will fall back onto the standard {@link JavaSerializer} if we detect either of
* these methods.
*
* Note that this class does not specialize the type on {@code Externalizable}. That is because if we fall back on the
* {@code JavaSerializer} it may have an {@code readResolve} method that returns an object of a different type.
*
* @author Robert DiFalco */
public class ExternalizableSerializer extends Serializer {
private ObjectMap javaSerializerByType;
private KryoObjectInput objectInput = null;
private KryoObjectOutput objectOutput = null;
public void write (Kryo kryo, Output output, Object object) {
JavaSerializer serializer = getJavaSerializerIfRequired(object.getClass());
if (serializer == null)
writeExternal(kryo, output, object);
else
serializer.write(kryo, output, object);
}
public Object read (Kryo kryo, Input input, Class type) {
JavaSerializer serializer = getJavaSerializerIfRequired(type);
if (serializer == null) return readExternal(kryo, input, type);
return serializer.read(kryo, input, type);
}
private void writeExternal (Kryo kryo, Output output, Object object) {
try {
((Externalizable)object).writeExternal(getObjectOutput(kryo, output));
} catch (Exception ex) {
throw new KryoException(ex);
}
}
private Object readExternal (Kryo kryo, Input input, Class type) {
try {
Externalizable object = (Externalizable)kryo.newInstance(type);
object.readExternal(getObjectInput(kryo, input));
return object;
} catch (Exception ex) {
throw new KryoException(ex);
}
}
private ObjectOutput getObjectOutput (Kryo kryo, Output output) {
if (objectOutput == null)
objectOutput = new KryoObjectOutput(kryo, output);
else
objectOutput.setOutput(output);
return objectOutput;
}
private ObjectInput getObjectInput (Kryo kryo, Input input) {
if (objectInput == null)
objectInput = new KryoObjectInput(kryo, input);
else
objectInput.setInput(input);
return objectInput;
}
/** Determines if this class requires the fall-back {@code JavaSerializer}. If the class does not require any specialized Java
* serialization features then null will be returned.
* @param type the type we wish to externalize
* @return a {@code JavaSerializer} if the type requires more than simple externalization. */
private JavaSerializer getJavaSerializerIfRequired (Class type) {
JavaSerializer javaSerializer = getCachedSerializer(type);
if (javaSerializer == null && isJavaSerializerRequired(type)) javaSerializer = new JavaSerializer();
return javaSerializer;
}
private JavaSerializer getCachedSerializer (Class type) {
if (javaSerializerByType == null) {
javaSerializerByType = new ObjectMap();
return null;
}
return javaSerializerByType.get(type);
}
private boolean isJavaSerializerRequired (Class type) {
return hasInheritableReplaceMethod(type, "writeReplace") || hasInheritableReplaceMethod(type, "readResolve");
}
/* find out if there are any pesky serialization extras on this class */
private static boolean hasInheritableReplaceMethod (Class type, String methodName) {
Method method = null;
Class current = type;
while (current != null) {
try {
method = current.getDeclaredMethod(methodName);
break;
} catch (NoSuchMethodException ex) {
current = current.getSuperclass();
}
}
return ((method != null) && (method.getReturnType() == Object.class));
}
}
© 2015 - 2024 Weber Informatics LLC | Privacy Policy