tutorial:codec
Differences
This shows you the differences between two versions of the page.
Both sides previous revisionPrevious revisionNext revision | Previous revision | ||
tutorial:codec [2024/05/28 09:35] – solidblock | tutorial:codec [2024/06/30 14:16] (current) – solidblock | ||
---|---|---|---|
Line 1: | Line 1: | ||
- | ====== | + | ====== |
===== What is a codec ===== | ===== What is a codec ===== | ||
- | A **codec**, | + | A **codec**, |
- | For example, loot tables are written in json forms in data packs, and a loaded as '' | + | For example, loot tables are written in json forms in data packs, and are loaded as '' |
- | Codec was introduced in 1.16, but has been in increasingly widely used in Minecraft since 1.20. For example, '' | + | Codec was introduced in 1.16, but has mainly |
- | ===== How to use a codec ===== | + | ===== Using a codec ===== |
- | When you serialize or deserialize, | + | When you serialize or deserialize, |
<code java> | <code java> | ||
// serializing a BlockPos | // serializing a BlockPos | ||
Line 25: | Line 25: | ||
</ | </ | ||
- | As seen in the code, the result is not directly '' | + | As seen in the code, the result is not directly '' |
<code java> | <code java> | ||
- | // get the result when succeed | + | // get the result when succeeded |
final NbtList nbtList = new NbtList(); | final NbtList nbtList = new NbtList(); | ||
nbtList.add(NbtInt.of(1)); | nbtList.add(NbtInt.of(1)); | ||
Line 41: | Line 41: | ||
</ | </ | ||
- | ===== How to write a codec ===== | + | There is a special type of '' |
+ | |||
+ | ===== Writing | ||
+ | ==== Vanilla existing codecs ==== | ||
+ | Mojang has already written many codecs for you. You can directly use them in some cases, and in complex codecs, they may be also useful. | ||
+ | |||
+ | For primitive types, codecs are stored as fields of '' | ||
+ | * '' | ||
+ | * '' | ||
+ | * '' | ||
+ | |||
+ | For classes belonging to Minecraft, codecs are stored as their fields. For example: | ||
+ | * '' | ||
+ | * '' | ||
+ | * '' | ||
+ | |||
+ | Besides, '' | ||
+ | * '' | ||
+ | * '' | ||
+ | * '' | ||
+ | |||
+ | > **Tips:** You can learn more about how to write codecs by seeing how vanilla codecs are written. | ||
==== Mapping existing codec ==== | ==== Mapping existing codec ==== | ||
- | For simple objects, you can directly convert it between those have existing codecs. Mojang provides codecs for all primitive types and common types. In this example, you want to create a codec for '' | + | For simple objects, you can directly convert it between those have existing codecs. Mojang provides codecs for all primitive types and common types. In this example, you want to create a codec for '' |
<code java> | <code java> | ||
- | Codec< | + | |
+ | | ||
</ | </ | ||
When decoding, the serialized from is converted into string through '' | When decoding, the serialized from is converted into string through '' | ||
- | If the serialized from is something that cannot be converted into a string, a codec result may be '' | + | If the serialized from is something that cannot be converted into a string, a codec result may be '' |
<code java> | <code java> | ||
Codec< | Codec< | ||
try { | try { | ||
- | return DataResult.success(new Identifier(s)); | + | |
+ | | ||
} catch (InvalidIdentifierException e) { | } catch (InvalidIdentifierException e) { | ||
return DataResult.error(() -> "The identifier is invalid:" | return DataResult.error(() -> "The identifier is invalid:" | ||
Line 65: | Line 88: | ||
</ | </ | ||
- | Note that this time we use '' | + | Note that this time we use '' |
<code java> | <code java> | ||
Codec< | Codec< | ||
try { | try { | ||
- | return DataResult.success(new Identifier(s)); | + | |
+ | | ||
} catch (InvalidIdentifierException e) { | } catch (InvalidIdentifierException e) { | ||
return DataResult.error(() -> "The identifier is invalid:" | return DataResult.error(() -> "The identifier is invalid:" | ||
Line 75: | Line 99: | ||
}, identifier -> identifier.toString()); | }, identifier -> identifier.toString()); | ||
</ | </ | ||
+ | |||
+ | ==== Record codec ==== | ||
+ | === Required fields === | ||
+ | Most objects are complicated, | ||
+ | <code java> | ||
+ | public record Student(String name, int id, Vec3d pos) { | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | To create a codec for this record, we should specify the fields codec, specify how to get fields from a '' | ||
+ | <code java> | ||
+ | public static final Codec< | ||
+ | Codec.STRING.fieldOf(" | ||
+ | Codec.INT.fieldOf(" | ||
+ | Vec3d.CODEC.fieldOf(" | ||
+ | ).apply(i, Student:: | ||
+ | </ | ||
+ | In this example, the method '' | ||
+ | |||
+ | Let's demonstrate the effect of the codec: | ||
+ | * '' | ||
+ | * '' | ||
+ | * '' | ||
+ | * '' | ||
+ | |||
+ | === Optional fields === | ||
+ | Sometimes all fields are not required. In the previous example, if the encoded result misses some fields, error will be thrown. To make the fields optional, use '' | ||
+ | <code java> | ||
+ | public static final Codec< | ||
+ | Codec.STRING.fieldOf(" | ||
+ | Codec.INT.optionalFieldOf(" | ||
+ | Vec3d.CODEC.fieldOf(" | ||
+ | ).apply(i, Student:: | ||
+ | </ | ||
+ | |||
+ | In this case, when decoding, when the field '' | ||
+ | * '' | ||
+ | * '' | ||
+ | * Of course, '' | ||
+ | * '' | ||
+ | |||
+ | Pay attention to the last example. When decoding, when an optional field has an invalid value, error will be thrown. Howevevr, in older Minecraft versions, when an optional field has an invalid value, the default value will be directly taken. | ||
+ | * In older Minecraft versions, '' | ||
+ | |||
+ | > **Note:** In current versions, you can also replace '' | ||
+ | |||
+ | If you do not provide a default value for '' | ||
+ | <code java> | ||
+ | public record Student(Optional< | ||
+ | public static final Codec< | ||
+ | Codec.STRING.optionalFieldOf(" | ||
+ | Codec.INT.fieldOf(" | ||
+ | Vec3d.CODEC.fieldOf(" | ||
+ | ).apply(i, Student:: | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | === Another from of field codec === | ||
+ | The codec can also be written like this: | ||
+ | <code java> | ||
+ | public record Student(String name, int id, Vec3d pos) { | ||
+ | public static final Codec< | ||
+ | Student:: | ||
+ | Codec.STRING.fieldOf(" | ||
+ | Codec.INT.fieldOf(" | ||
+ | Vec3d.CODEC.fieldOf(" | ||
+ | )); | ||
+ | } | ||
+ | </ | ||
+ | Writing like this may be simpler. You need to specify the method to convert fields to an objects at first ('' | ||
+ | ==== Dispatching codec ==== | ||
+ | Some objects may not be in fixed structures, but have variant types, each of which, is in a unique structure. Therefore, dispatching codecs are used. | ||
+ | |||
+ | When encoding, the codec gets the type from the object, encode it according to a codec corresponding to the " | ||
+ | |||
+ | Let's take this example: '' | ||
+ | <code java> | ||
+ | public interface SchoolMember { | ||
+ | record Student(String name, int id) implements SchoolMember {} | ||
+ | record Teacher(String name, String subject, int id) implements SchoolMember {} | ||
+ | record Staff(String name, String department, Identifier id) implements SchoolMember {} | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | It's easy to know that each type can be created a specific codec (note that it is '' | ||
+ | <code java> | ||
+ | public interface SchoolMember { | ||
+ | record Student(String name, int id) implements SchoolMember { | ||
+ | public static final MapCodec< | ||
+ | } | ||
+ | // Codecs of other two types are omitted here. | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | We also need to specify a serializable " | ||
+ | <code java> | ||
+ | public interface SchoolMember { | ||
+ | @NotNull Type getType(); | ||
+ | | ||
+ | enum Type implements StringIdentifiable { | ||
+ | STUDENT(" | ||
+ | public static final Codec< | ||
+ | private final String name; | ||
+ | |||
+ | Type(String name) { | ||
+ | this.name = name; | ||
+ | } | ||
+ | |||
+ | @Override | ||
+ | public String asString() { | ||
+ | return name; | ||
+ | } | ||
+ | } | ||
+ | |||
+ | record Student(String name, int id) implements SchoolMember { | ||
+ | public static final Codec< | ||
+ | |||
+ | @Override | ||
+ | public @NotNull Type getType() { | ||
+ | return Type.STUDENT; | ||
+ | } | ||
+ | } | ||
+ | |||
+ | // The other two types are omitted here. | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | Now we specified how to get the type from the object. However, it is also required to specify how to get the codec from the type. It can be achieved in many ways, such as storing fields in the type object, or using a map. But in this example, for simplicity, we just use a simple " | ||
+ | <code java> | ||
+ | Codec< | ||
+ | case STAFF -> Staff.CODEC; | ||
+ | case STUDENT -> Student.CODEC; | ||
+ | case TEACHER -> Teacher.CODEC; | ||
+ | }); | ||
+ | </ | ||
+ | |||
+ | If you need to specify another name of the field, add a string as a first parameter of '' | ||
+ | |||
+ | Let's see the effect our example: | ||
+ | * '' | ||
+ | * '' | ||
+ | |||
+ | > **Note:** Actually, in practice, the types can be more complicated. Therefore, you may use registry for types, such as vanilla '' | ||
+ | |||
+ | ===== Packet codec ===== | ||
+ | A **packet codec**, different from codec, converts between objects and binery packets. It is sometimes similar to codec, and also used in many cases such as item components, but for complex objects, it uses //tuples// instead of //maps//. Packet codecs for primitive types are stored in '' | ||
+ | <code java> | ||
+ | public record Student(String name, int id, Vec3d pos) { | ||
+ | public static final PacketCodec< | ||
+ | PacketCodecs.STRING, | ||
+ | PacketCodecs.INTEGER, | ||
+ | PacketCodec.of( | ||
+ | // encoder: writing to the packet | ||
+ | (value, buf) -> buf.writeDouble(value.x).writeDouble(value.y).writeDouble(value.z), | ||
+ | // decoder: reading the packet | ||
+ | buf -> new Vec3d(buf.readDouble(), | ||
+ | ), Student:: | ||
+ | Student:: | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | > **Note:** Besides '' |
tutorial/codec.1716888952.txt.gz · Last modified: 2024/05/28 09:35 by solidblock