title: Xlang Serialization sidebar_position: 2 id: xlang_serialization 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™ xlang serialization is the Java wire mode for payloads that must be read by Python, Rust, Go, JavaScript/TypeScript, C++, C#, Swift, Dart, Scala, Kotlin, or another non-Java Fory runtime. Java defaults to xlang mode with compatible schema evolution, but examples set the mode explicitly so the payload contract is visible in code.
Use one long-lived Fory or ThreadSafeFory instance per configuration. Creating a runtime is expensive because Fory caches type metadata and generated serializers.
import org.apache.fory.Fory; Fory fory = Fory.builder() .withXlang(true) .requireClassRegistration(true) .withRefTracking(true) .build();
withRefTracking(true) is required only when the cross-language data model includes shared object identity or cycles. Disable it for value-shaped schemas.
Use Native Serialization instead when every writer and reader is Java and the payload should preserve Java-specific object behavior.
Types must be registered with consistent IDs or names across all languages. Fory supports two registration methods.
public record Person(String name, int age) {} // Numeric ID registration is compact and fast. fory.register(Person.class, 1); Person person = new Person("Alice", 30); byte[] bytes = fory.serialize(person); // bytes can be deserialized by Python, Rust, Go, etc.
Benefits: faster serialization and smaller binary size.
Trade-off: every service must coordinate IDs so the same logical type uses the same number.
public record Person(String name, int age) {} // Namespace/type-name registration is easier to coordinate across teams. fory.register(Person.class, "example", "Person"); Person person = new Person("Alice", 30); byte[] bytes = fory.serialize(person); // bytes can be deserialized by Python, Rust, Go, etc.
Benefits: less risk of numeric ID conflicts and easier management across independently owned services.
Trade-off: the payload includes string identity, so it is larger than ID-based registration.
The Java API also supports a single string type name, such as fory.register(Person.class, "example.Person"). Use the same logical identity on every runtime.
import org.apache.fory.Fory; public record Person(String name, int age) {} public class Example { public static void main(String[] args) { Fory fory = Fory.builder() .withXlang(true) .withRefTracking(true) .build(); // Register with the same logical name used by Python. fory.register(Person.class, "example.Person"); Person person = new Person("Bob", 25); byte[] bytes = fory.serialize(person); // Send bytes to Python by your service transport. } }
import pyfory from dataclasses import dataclass @dataclass class Person: name: str age: pyfory.Int32 fory = pyfory.Fory(xlang=True, ref=True) # Register with the same name as Java. fory.register_type(Person, typename="example.Person") person = fory.deserialize(bytes_from_java) print(f"{person.name}, {person.age}") # Output: Bob, 25
Xlang mode supports circular and shared references when reference tracking is enabled:
public class Node { public String value; public Node next; public Node parent; } Fory fory = Fory.builder() .withXlang(true) .withRefTracking(true) .build(); fory.register(Node.class, "example.Node"); Node node1 = new Node(); node1.value = "A"; Node node2 = new Node(); node2.value = "B"; node1.next = node2; node2.parent = node1; byte[] bytes = fory.serialize(node1); // Python/Rust/Go can correctly deserialize this with circular references preserved
Not all Java types have equivalents in other languages. When using xlang mode:
int, long, double, String) for maximum compatibility.List, Map, Set) instead of language-specific collections.Float16, BFloat16, Float16List, BFloat16List) for 16-bit float payloads.Float16[], BFloat16[], Float16List, and BFloat16List as list<T> carriers by default; use @ArrayType when the schema must be array<float16> or array<bfloat16>.Optional, BigDecimal, and EnumSet unless every target runtime has an agreed mapping.Java primitive arrays are dense array<T> carriers, except plain byte[], which defaults to bytes. General Java collections and Fory primitive-list carriers such as Int32List, Float16List, and BFloat16List use list<T> unless the field has explicit @ArrayType metadata.
| Fory schema | Java field shape |
|---|---|
list<int32> | List<Integer> or Int32List |
array<bool> | boolean[] |
array<int8> | @Int8Type byte[] type-use |
array<int16> | short[] |
array<int32> | int[] |
array<int64> | long[] |
array<uint8> | @UInt8Type byte[] type-use |
array<uint16> | @UInt16Type short[] type-use |
array<uint32> | @UInt32Type int[] type-use |
array<uint64> | @UInt64Type long[] type-use |
array<float16> | Float16Array or @Float16Type short[] |
array<bfloat16> | BFloat16Array or @BFloat16Type short[] |
array<float32> | float[] |
array<float64> | double[] |
Prefer type-use syntax for primitive-array annotations:
private @UInt32Type int[] ids; private @BFloat16Type short[] values;
public record UserData( String name, // compatible int age, // compatible List<String> tags, // compatible Map<String, Integer> scores // compatible ) {}
public record UserData( Optional<String> name, // not cross-language compatible BigDecimal balance, // limited support EnumSet<Status> statuses // Java-specific collection ) {}
Xlang mode has additional overhead compared to Java native mode:
For best performance:
withRefTracking(false))withXlang(false)) when only Java serialization is needed