Unfortunately, with new collection types comes incompatibility with normal serialization frameworks. Jackson, arguably the most popular JSON serialization framework for Java, is unable to deserialize Eclipse Collections types out-of-the-box. For this purpose, there is now a Jackson module supporting most Eclipse Collections types directly (including primitive collections).
In this article we will cover the basics of Eclipse Collections, Jackson and that module.
Eclipse Collections replaces many of the familiar container types from the standard library with new and improved versions. For this article, we will look at three variants of the List
type:
import org.eclipse.collections.api.list.ImmutableList;
import org.eclipse.collections.api.list.MutableList;
import org.eclipse.collections.api.list.primitive.IntList;
import org.eclipse.collections.impl.factory.Lists;
import org.eclipse.collections.impl.factory.primitive.IntLists;
MutableList<String> mutable = Lists.mutable.of("foo", "bar");
ImmutableList<String> immutable = Lists.immutable.of("foo", "bar");
IntList intList = IntLists.immutable.of(1, 2, 3);
MutableList
behaves a lot like java.util.List
, but offers additional stream-like API we will not go into detail about here. Every MutableList
also implements java.util.List
, so you can pass any MutableList
to APIs that still use the standard Java collections.
ImmutableList
does not extend java.util.List
and - as its name suggests - does not offer any methods for mutation such as add
or remove
. Instead you can create copies of these collections with elements added or removed. If you wish to use these collections with traditional java.util.List
APIs you can use the castToList()
method on ImmutableList
, which will return an instance of java.util.List
that will throw exceptions on mutation operations like remove
.
IntList
is a special type of list that offers lightweight APIs for working with primitives. Similar collection types exist for all other primitives. The problem with "normal" collections and primitives is that using a List<Integer>
or similar type will always carry additional overhead both in terms of more memory use and more CPU time due to indirection and additional garbage collection load. Because of this, for performance-critical operations, a dedicated IntList
type is superior.
Eclipse Collections is hardly the first library to provide primitive collections but combined with its handy API it makes for arguably the best library of its kind. Using Eclipse primitive collections you can have the performance of primitive arrays with the ease-of-use of the Eclipse Collections API.
For all the primitive collection types there also exist mutable and immutable versions (MutableIntList
, ImmutableIntList
). They behave much the same as their object-based counterparts so we will not cover them here.
Eclipse Collections is split into two parts: The API and the implementation of that API. Since the implementation depends on the API, we will only have to add that as a dependency to our Maven project. See the latest release for the details on the Maven or Gradle coordinates.
Jackson is a library for transforming your data structures from and to JSON and other serialized forms. According to mvnrepository.com it is the most popular framework for this purpose.
Jackson's architecture consists of three parts: A core that is responsible for writing streams of data (typically JSON), databind which is responsible for the basic object serialization API, and various Jackson modules that each handle serialization of some special types. Adding Jackson to our project is simple by using the Maven or Gradle coordinates for the latest release of jackson-databind and jackson-datatype-eclipse-collections.
To start out with Jackson we create an ObjectMapper
:
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.datatype.eclipsecollections.EclipseCollectionsModule;
ObjectMapper mapper = new ObjectMapper().registerModule(new EclipseCollectionsModule());
We are now ready to serialize data:
public class MyClass {
// The fields of this class must be public, or must have getters and
// setters. For simplicity we will stick with public fields here.
public String text;
public int value;
}
MyClass object = new MyClass();
object.text = "foo";
object.value = 42;
String json = mapper.writeValueAsString(object);
// json = {"text":"foo","value":42}
MyClass deserialized = mapper.readValue(json, MyClass.class);
Now we are ready to serialize our Eclipse Collections! Lets start with a simple example:
public class EclipseClass {
public ImmutableList<String> strings;
}
EclipseClass object = new EclipseClass();
object.strings = Lists.immutable.of("foo", "bar");
String json = mapper.writeValueAsString(object);
// json = {"strings":["foo","bar"]}
EclipseClass deserialized = mapper.readValue(json, EclipseClass.class);
As you can see, the list was serialized to a JSON array, as expected. Lets try with a primitive collection:
public class EclipseClassInt {
public IntList ints;
}
EclipseClassInt object = new EclipseClassInt();
object.ints = IntLists.immutable.of(1, 2, 3);
String json = mapper.writeValueAsString(object);
// json = {"ints":[1,2,3]}
EclipseClassInt deserialized = mapper.readValue(json, EclipseClassInt.class);
Using the jackson-datatype-eclipse-collections
module primitive collections are treated just like normal collections.
Sometimes you do not want to write your own type wrapper but want to serialize collections directly. Because of generics, this takes some extra work to deserialize in Jackson:
MutableList<String> mutable = Lists.mutable.of("foo", "bar");
String json = mapper.writeValueAsString(mutable);
// json = ["foo","bar"]
MutableList<String> deserialized = mapper.readValue(json, new TypeReference<MutableList<String>>() {});
Because Java does not have a syntax like MutableList<String>.class
, we have to stick with the more convoluted Jackson type references. Do not attempt to use MutableList.class
instead - you will get an error that does not immediately make clear what the issue is.
For primitive collections, because they do not use generics, using .class
is possible:
IntList intList = IntLists.immutable.of(1, 2, 3);
String json = mapper.writeValueAsString(intList);
// json = [1,2,3]
IntList deserialized = mapper.readValue(json, IntList.class);
The Eclipse Collections module for Jackson attempts to be consistent in the serialization of primitive collections with their array counterparts. IntList
will serialize like an int[]
would, CharList
like a char[]
would and so on. For char
and byte
this can make for a more compact representation. ByteList
is serialized as base64 strings (or a binary type for non-JSON formats):
ByteList byteList = ByteLists.immutable.of(new byte[]{ 73, 20, 32, 23 });
String json = mapper.writeValueAsString(byteList);
// json = "SRQgFw=="
Similarly, CharList
will just be serialized to a JSON string:
CharList charList = CharLists.immutable.of('f', 'o', 'o');
String json = mapper.writeValueAsString(charList);
// json = "foo"
For CharList
, this feature can be turned off in Jackson:
CharList charList = CharLists.immutable.of('f', 'o', 'o');
String json = mapper.writer(SerializationFeature.WRITE_CHAR_ARRAYS_AS_JSON_ARRAYS)
.writeValueAsString(charList);
// json = ["f","o","o"]