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/06/30 12:18] – [Dispatching codec] solidblock | tutorial:codec [2024/06/30 14:16] (current) – solidblock | ||
---|---|---|---|
Line 1: | Line 1: | ||
- | ====== | + | ====== |
===== What is a codec ===== | ===== What is a codec ===== | ||
- | A **codec**, introduced in Java Edition 1.16, is a specification of conversion between any type of object (such as '' | + | A **codec**, introduced in Java Edition 1.16, is a specification of conversion between any type of object (such as '' |
- | For example, loot tables are written in json forms in data packs, and are 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 mainly been increasingly widely used in Minecraft since 1.20. For example, '' | + | Codec was introduced in 1.16, but has mainly been increasingly widely used in Minecraft since 1.20. For example, '' |
===== Using a codec ===== | ===== Using a codec ===== | ||
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: | ||
</ | </ | ||
- | There is a special type of '' | + | There is a special type of '' |
- | ===== How to write a codec ===== | + | ===== Writing |
==== Vanilla existing codecs ==== | ==== 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. | 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. | ||
Line 57: | Line 57: | ||
* '' | * '' | ||
- | Besides, '' | + | Besides, '' |
* '' | * '' | ||
* '' | * '' | ||
Line 63: | Line 63: | ||
> **Tips:** You can learn more about how to write codecs by seeing how vanilla codecs are written. | > **Tips:** You can learn more about how to write codecs by seeing how vanilla codecs are written. | ||
+ | |||
==== Mapping existing codec ==== | ==== Mapping existing codec ==== | ||
Line 101: | Line 102: | ||
==== Record codec ==== | ==== Record codec ==== | ||
=== Required fields === | === Required fields === | ||
- | Most objects are complicated, | + | Most objects are complicated, |
<code java> | <code java> | ||
public record Student(String name, int id, Vec3d pos) { | public record Student(String name, int id, Vec3d pos) { | ||
Line 167: | Line 168: | ||
} | } | ||
</ | </ | ||
- | Writing like this may be simpler. You need to specify the method to convert fields to an objects at first ('' | + | Writing like this may be simpler. You need to specify the method to convert fields to an objects at first ('' |
==== Dispatching codec ==== | ==== 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, | + | 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: '' | Let's take this example: '' | ||
Line 180: | Line 183: | ||
</ | </ | ||
- | It's easy to know that each type can be created a specific codec: | + | It's easy to know that each type can be created a specific codec (note that it is '' |
<code java> | <code java> | ||
public interface SchoolMember { | public interface SchoolMember { | ||
record Student(String name, int id) implements SchoolMember { | record Student(String name, int id) implements SchoolMember { | ||
- | public static final Codec< | + | public static final MapCodec< |
} | } | ||
- | // other types similar | + | // Codecs of other two types are omitted here. |
} | } | ||
</ | </ | ||
Line 219: | Line 222: | ||
} | } | ||
- | // other two types similar. | + | // 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 | ||
+ | <code java> | ||
+ | public record Student(String name, int id, Vec3d pos) { | ||
+ | | ||
+ | PacketCodecs.STRING, | ||
+ | PacketCodecs.INTEGER, | ||
+ | PacketCodec.of( | ||
+ | | ||
+ | (value, buf) -> buf.writeDouble(value.x).writeDouble(value.y).writeDouble(value.z), | ||
+ | // decoder: reading the packet | ||
+ | buf -> new Vec3d(buf.readDouble(), | ||
+ | ), Student:: | ||
+ | Student:: | ||
} | } | ||
</ | </ | ||
- | Now we specified how to get the type from the object. | + | > **Note:** Besides '' |
tutorial/codec.1719749936.txt.gz · Last modified: 2024/06/30 12:18 by solidblock