SubZero provides dead easy Hazelcast - Kryo integration. No coding required.
Kryo is a popular serialization library. It's super-fast yet easy-to-use.
It does not pollute your domain model and it can even serialize classes
which are not marked as Serializable
.
Hazelcast has no out-of-the box support for Kryo. It's rather easy to integrate it, however it means everyone has to write the same code and face the same bugs.
SubZero aims to make Kryo - Hazelcast integration as simple as possible.
SubZero will completely replace Java serialization. Hazelcast internal serializers will still take precedence.
Insert this snippet into your Hazelcast configuration XML:
<serialization>
<serializers>
<global-serializer override-java-serialization="true">
info.jerrinot.subzero.Serializer
</global-serializer>
</serializers>
</serialization>
Config config = new Config();
SubZero.useAsGlobalSerializer(config);
HazelcastInstance hz = Hazelcast.newHazelcastInstance(config);
In this mode Hazelcast will use SubZero for selected classes only.
<serialization>
<serializers>
<serializer type-class="some.package.Foo"
class-name="info.jerrinot.subzero.Serializer"/>
<serializer type-class="some.package.Bar"
class-name="info.jerrinot.subzero.Serializer"/>
</serializers>
</serialization>
Config config = new Config();
SubZero.useForClasses(config, Foo.class, Bar.class);
HazelcastInstance hz = Hazelcast.newHazelcastInstance(config);
All cluster members have to use SubZero for the same types and the types have to be declared in the same order. As of Hazelcast 3.7 programmatic configuration will result in somewhat higher performance - this is given by a limitation of Hazelcast declarative configuration API. It should be fixed Hazelcast 3.8
SubZero is available in Maven Central. Just insert this snippet into pom.xml and you are ready to roll!
<dependency>
<groupId>info.jerrinot</groupId>
<artifactId>subzero-all</artifactId>
<version>0.9</version>
</dependency>
This version has all dependencies packaged inside. You can also use a version with regular dependencies:
<dependency>
<groupId>info.jerrinot</groupId>
<artifactId>subzero-core</artifactId>
<version>0.9</version>
</dependency>
- System property
subzero.buffer.size.kb
sets buffer size for Kryo. Default value: 16KB - System property
subzero.base.type.id
sets base for auto-generated type id - System property
subzero.referenceresolver.class
sets the ReferenceResolver implementation. Default value:com.esotericsoftware.kryo.util.MapReferenceResolver
SubZero can use custom Kryo serializers.
Just create a file subzero-serializers.properties
and have it on a classpath of your project. SubZero expects the property file to have the following format:
some.package.YouDomainClass=other.package.KryoSerializer
The simple approach works fine in most cases, but sometimes you do not know
domain classnames up front - for example when the class is
created by a factory - think of Collections::unmodifiableList
In this case it's OK to have just the Kryo serializers in property file. For example:
my.package.KryoSerializerClass
Subzero expects the serializer to have a method registerSerializers
which accepts an instance of Kryo
as its only argument.
It's up to the serializer to register itself into Kryo. This approach works for most serializer from this project. Actually serializers from this project are considered to be well-known and it's ok to use just a classname without package in the property file. For example:
UnmodifiableCollectionsSerializer
ArraysAsListSerializer
Kryo uses FieldSerializer by default. Sometimes you want to change it.
For example when you want support adding new fields then you have to use CompatibleFieldSerializer.
You can do this by using the key defaultSerializer
in the property file.
Example:
defaultSerializer=com.esotericsoftware.kryo.serializers.CompatibleFieldSerializer
This mechanism also understands the concept of well-known packages: Serializers from the Kryo project itself and Magro serializers can be referenced just by a simple classname, no need to specify package. This is handy when using the Subzero-all version which has the Kryo serializers relocated into another package.
SubZero aims to provide the simplest possible way to hook Kryo serialization into Hazelcast.
Default SubZero serializer implementation uses auto-generated class type IDs and relies on a serializer registration order. This means all your cluster members have to use the same order in Hazelcast serializer configuration. This can be somewhat fragile. You can make it more robust by subclassing Serializer and returning a fixed class ID:
public class HashMapSerializerExample extends AbstractTypeSpecificUserSerializer<HashMap> {
public HashMapSerializerExample() {
super(HashMap.class);
}
/**
* TypeId has to be a unique for each registered serializer.
*
* @return TypeId of the class serialized by this serializer
*/
@Override
public int getTypeId() {
return 10000;
}
}
SubZero is continuously tested with Hazelcast 3.6, 3.7, 3.8, 3.9, 3.10, 3.11 and 3.12-SNAPSHOT.
- More serialization strategies. Currently Kryo is the only supported strategy. I would like to add Fast Serialization
- Serializer Generator - SubZero could generate highly-optimized serializer for simple classes
- AutoPortable - serialize an ordinary class as it implemented Portable interface
- Ashok Koyi reported a first bug and also came up with a solution.
- Will Neild came up with an idea to register custom Kryo serializers and contributed to other parts as well.
- Adrian contributed support for custom reference resolvers.
This is a community project not affiliated with the Hazelcast project.