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 11:02] – [How to write a 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**, | + | A **codec**, |
- | 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 been 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: | ||
</ | </ | ||
- | 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 118: | Line 119: | ||
Let's demonstrate the effect of the codec: | Let's demonstrate the effect of the codec: | ||
- | * '' | + | * '' |
- | * '' | + | * '' |
- | * '' | + | * '' |
* '' | * '' | ||
=== Optional fields === | === Optional fields === | ||
- | Sometimes all fields are not required. In the preview field, if the encoded result misses some fields, error will be thrown. | + | Sometimes all fields are not required. In the previous example, if the encoded result misses some fields, error will be thrown. |
<code java> | <code java> | ||
public static final Codec< | public static final Codec< | ||
Line 134: | Line 135: | ||
In this case, when decoding, when the field '' | In this case, when decoding, when the field '' | ||
- | * '' | + | * '' |
- | * '' | + | * '' |
- | * Of course, '' | + | * 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. | 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, '' | + | * In older Minecraft versions, '' |
- | In current versions, you can also replace '' | + | > **Note: |
If you do not provide a default value for '' | If you do not provide a default value for '' | ||
Line 167: | Line 168: | ||
} | } | ||
</ | </ | ||
- | Writing like this may be simpler. You should | + | 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 have different | + | Some objects may not be in fixed structures, but have variant |
+ | |||
+ | When encoding, | ||
+ | |||
+ | 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 | ||
+ | <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.1719745330.txt.gz · Last modified: 2024/06/30 11:02 by solidblock