title: Java Serialization Guide sidebar_position: 0 id: serialization_index license: | Licensed to the Apache Software Foundation (ASF) under one or more contributor license agreements. See the NOTICE file distributed with this work for additional information regarding copyright ownership. The ASF licenses this file to You under the Apache License, Version 2.0 (the “License”); you may not use this file except in compliance with the License. You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Apache Fory™ provides blazingly fast Java object serialization with JIT compilation and zero-copy techniques. Java supports both xlang mode and native mode. Xlang mode is the default cross-language wire format and uses compatible schema evolution. Native mode is the Java-only wire format for same-language object serialization, JDK serialization replacement behavior, framework replacement, and Java-native object graph features.
writeObject/readObject/writeReplace/readResolve/readObjectNoData/Externalizable<dependency> <groupId>org.apache.fory</groupId> <artifactId>fory-core</artifactId> <version>1.2.0</version> </dependency>
implementation("org.apache.fory:fory-core:1.2.0")
On JDK25+, open java.lang.invoke to Fory. Use ALL-UNNAMED when Fory is on the classpath:
--add-opens=java.base/java.lang.invoke=ALL-UNNAMED
Use the Fory core module name when Fory is on the module path:
--add-opens=java.base/java.lang.invoke=org.apache.fory.core
Note that Fory creation is not cheap, the Fory instances should be reused between serializations instead of creating it every time. You should keep Fory as a static global variable, or instance variable of some singleton object or limited objects.
import java.util.List; import java.util.Arrays; import org.apache.fory.*; import org.apache.fory.config.*; public class Example { public static void main(String[] args) { SomeClass object = new SomeClass(); // Note that Fory instances should be reused between // multiple serializations of different objects. Fory fory = Fory.builder() .withXlang(true) .requireClassRegistration(true) .build(); // Registering types can reduce class name serialization overhead, but not mandatory. // If class registration enabled, all custom types must be registered. // Registration order must be consistent if id is not specified fory.register(SomeClass.class); byte[] bytes = fory.serialize(object); System.out.println(fory.deserialize(bytes)); } }
import org.apache.fory.*; import org.apache.fory.config.*; public class Example { public static void main(String[] args) { SomeClass object = new SomeClass(); ThreadSafeFory fory = Fory.builder() .withXlang(true) .buildThreadSafeFory(); fory.register(SomeClass.class, 1); byte[] bytes = fory.serialize(object); System.out.println(fory.deserialize(bytes)); } }
import org.apache.fory.*; import org.apache.fory.config.*; public class Example { private static final ThreadSafeFory fory = Fory.builder() .withXlang(true) .buildThreadSafeFory(); static { fory.register(SomeClass.class, 1); } public static void main(String[] args) { SomeClass object = new SomeClass(); byte[] bytes = fory.serialize(object); System.out.println(fory.deserialize(bytes)); } }
Use xlang mode for cross-language payloads and schemas shared with non-Java implementations. It is the default Java wire mode, and Java examples that use it set .withXlang(true) explicitly so the mode choice is visible.
Use native mode for Java-only traffic. Native mode is selected with .withXlang(false) and owns Java-specific object behavior such as JDK serialization hooks, Externalizable, dynamic object graphs, object copy, and Java native-mode zero-copy buffers. It is optimized for the JVM type system and supports a broader Java object surface than xlang mode. Compatible mode is enabled by default. Set .withCompatible(false) only when every reader and writer uses the same class schema and you want faster serialization and smaller size. If you are replacing JDK serialization, Kryo, FST, Hessian, or Java-only Protocol Buffers payloads, start with native mode.
See Native Serialization for Java-only serialization details and Xlang Serialization for Java xlang registration and interoperability rules.
Fory provides two thread-safe Fory instance styles:
buildThreadSafeForyThis is the default choice. It uses a fixed-size shared ThreadPoolFory sized to 4 * availableProcessors() and is the preferred instance form for virtual-thread workloads:
ThreadSafeFory fory = Fory.builder() .withXlang(true) .withRefTracking(false) .withAsyncCompilation(true) .buildThreadSafeFory();
See more details in Virtual Threads.
Use buildThreadLocalFory() only when you explicitly want one Fory instance per long-lived platform thread, or when you want to pin that choice regardless of JDK version:
ThreadSafeFory fory = Fory.builder() .withXlang(true) .buildThreadLocalFory(); fory.register(SomeClass.class, 1); byte[] bytes = fory.serialize(object); System.out.println(fory.deserialize(bytes));
buildThreadSafeForyPoolUse buildThreadSafeForyPool(poolSize) when you want to set that fixed shared pool size explicitly. It eagerly creates poolSize Fory instances, keeps them in shared fixed slots, and then lets any caller borrow one through a thread-agnostic fast path. Calls only block when every pooled instance is already in use; the pool does not key cached instances by thread identity:
ThreadSafeFory fory = Fory.builder() .withXlang(true) .withRefTracking(false) .withAsyncCompilation(true) .buildThreadSafeForyPool(poolSize);
// Single-thread Fory Fory fory = Fory.builder() .withXlang(true) .withRefTracking(false) .withAsyncCompilation(true) .build(); // Thread-safe Fory (thread-safe Fory backed by a pool of Fory instances) ThreadSafeFory fory = Fory.builder() .withXlang(true) .withRefTracking(false) .withAsyncCompilation(true) .buildThreadSafeFory(); // Explicit thread-local Fory instance ThreadSafeFory threadLocalFory = Fory.builder() .withXlang(true) .buildThreadLocalFory();
@ForyField, @Ignore, integer encoding annotations, serializeEnumByName, and @ForyEnumId@ForyStruct