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 10:26] – [Record 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 ==== | ||
+ | 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 ==== | ||
Line 50: | Line 69: | ||
<code java> | <code java> | ||
- | Codec< | + | |
+ | | ||
</ | </ | ||
Line 60: | Line 80: | ||
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 71: | Line 92: | ||
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 79: | Line 101: | ||
==== Record codec ==== | ==== Record codec ==== | ||
- | Most objects are complicated, | + | === Required fields === |
+ | 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 89: | Line 112: | ||
public static final Codec< | public static final Codec< | ||
Codec.STRING.fieldOf(" | Codec.STRING.fieldOf(" | ||
- | Codec.INT.fieldOf(" | + | Codec.INT.fieldOf(" |
Vec3d.CODEC.fieldOf(" | Vec3d.CODEC.fieldOf(" | ||
).apply(i, Student:: | ).apply(i, Student:: | ||
</ | </ | ||
In this example, the method '' | 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 ==== | ==== 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.1719743202.txt.gz · Last modified: 2024/06/30 10:26 by solidblock