数据组件(Data Components)
数据组件(Data Components)
数据组件是存储在ItemStack
中的键值对,用于存储物品堆叠的数据。每个数据(如烟花爆炸效果或工具属性)都以实际对象的形式存储在堆叠中,使得这些值可以直接访问和操作,而不需要动态转换通用的编码实例(例如CompoundTag
或JsonElement
)。
DataComponentType
每个数据组件都有一个关联的DataComponentType<T>
,其中T
是组件值的类型。DataComponentType
表示一个键,用于引用存储的组件值,并包含一些编解码器(Codec),用于处理磁盘和网络的读写操作(如果需要)。
现有的组件列表可以在DataComponents
中找到。
创建自定义数据组件
与DataComponentType
关联的组件值必须实现hashCode
和equals
方法,并且在存储时应被视为不可变。
注意:组件值可以很容易地使用record
实现。record
字段是不可变的,并且自动实现了hashCode
和equals
。
// 使用 record 的示例
public record ExampleRecord(int value1, boolean value2) {}
// 使用 class 的示例
public class ExampleClass {
private final int value1;
// 可以是可变的,但在使用时需要小心
private boolean value2;
public ExampleClass(int value1, boolean value2) {
this.value1 = value1;
this.value2 = value2;
}
@Override
public int hashCode() {
return Objects.hash(this.value1, this.value2);
}
@Override
public boolean equals(Object obj) {
if (obj == this) {
return true;
} else {
return obj instanceof ExampleClass ex
&& this.value1 == ex.value1
&& this.value2 == ex.value2;
}
}
}
标准的DataComponentType
可以通过DataComponentType#builder
创建,并使用DataComponentType.Builder#build
构建。构建器包含三个设置:persistent
、networkSynchronized
和cacheEncoding
。
-
persistent
:指定用于将组件值读写到磁盘的Codec
。 -
networkSynchronized
:指定用于通过网络读写组件的StreamCodec
。如果未指定networkSynchronized
,则将使用persistent
中提供的Codec
作为StreamCodec
。 -
cacheEncoding
:缓存Codec
的编码结果,以便在组件值未更改时使用缓存值。这应该仅在组件值很少或从不更改时使用。
警告:构建器中必须提供persistent
或networkSynchronized
,否则会抛出NullPointerException
。如果不需要通过网络发送数据,则将networkSynchronized
设置为StreamCodec#unit
,并提供默认的组件值。
DataComponentType
是注册对象,必须注册。
示例代码
// 使用 ExampleRecord(int, boolean)
// 下面只应使用一个 Codec 和/或 StreamCodec
// 提供多个是为了示例
// 基础 Codec
public static final Codec<ExampleRecord> BASIC_CODEC = RecordCodecBuilder.create(instance ->
instance.group(
Codec.INT.fieldOf("value1").forGetter(ExampleRecord::value1),
Codec.BOOL.fieldOf("value2").forGetter(ExampleRecord::value2)
).apply(instance, ExampleRecord::new)
);
public static final StreamCodec<ByteBuf, ExampleRecord> BASIC_STREAM_CODEC = StreamCodec.composite(
ByteBufCodecs.INT, ExampleRecord::value1,
ByteBufCodecs.BOOL, ExampleRecord::value2,
ExampleRecord::new
);
// 如果不需要通过网络发送数据,使用 Unit StreamCodec
public static final StreamCodec<ByteBuf, ExampleRecord> UNIT_STREAM_CODEC = StreamCodec.unit(new ExampleRecord(0, false));
// 在另一个类中
// 专门的 DeferredRegister.DataComponents 简化了数据组件注册,并避免了 `DataComponentType.Builder` 在 `Supplier` 中的一些泛型推断问题
public static final DeferredRegister.DataComponents REGISTRAR = DeferredRegister.createDataComponents(Registries.DATA_COMPONENT_TYPE, "examplemod");
public static final DeferredHolder<DataComponentType<?>, DataComponentType<ExampleRecord>> BASIC_EXAMPLE = REGISTRAR.registerComponentType(
"basic",
builder -> builder
// 用于读写数据的 Codec
.persistent(BASIC_CODEC)
// 用于通过网络读写数据的 Codec
.networkSynchronized(BASIC_STREAM_CODEC)
);
/// 组件不会保存到磁盘
public static final DeferredHolder<DataComponentType<?>, DataComponentType<ExampleRecord>> TRANSIENT_EXAMPLE = REGISTRAR.registerComponentType(
"transient",
builder -> builder.networkSynchronized(BASIC_STREAM_CODEC)
);
// 不会通过网络同步数据
public static final DeferredHolder<DataComponentType<?>, DataComponentType<ExampleRecord>> NO_NETWORK_EXAMPLE = REGISTRAR.registerComponentType(
"no_network",
builder -> builder
.persistent(BASIC_CODEC)
// 注意我们在这里使用 Unit StreamCodec
.networkSynchronized(UNIT_STREAM_CODEC)
);
组件映射(Component Map)
所有数据组件都存储在DataComponentMap
中,使用DataComponentType
作为键,对象作为值。DataComponentMap
的功能类似于只读的Map
。因此,可以使用#get
方法根据DataComponentType
获取条目,或者如果不存在则提供默认值(通过#getOrDefault
)。
// 对于某个 DataComponentMap map
// 如果组件存在,获取染料颜色
// 否则返回 null
@Nullable
DyeColor color = map.get(DataComponents.BASE_COLOR);
修补的组件映射(PatchedDataComponentMap)
由于默认的DataComponentMap
仅提供基于读的操作,基于写的操作通过子类PatchedDataComponentMap
支持。这包括#set
设置组件的值或#remove
完全移除它。
PatchedDataComponentMap
使用原型和补丁映射存储更改。原型是一个DataComponentMap
,包含此映射应具有的默认组件及其值。补丁映射是一个DataComponentType
到Optional
值的映射,包含对默认组件的更改。
// 对于某个 PatchedDataComponentMap map
// 将基础颜色设置为白色
map.set(DataComponents.BASE_COLOR, DyeColor.WHITE);
// 移除基础颜色
// - 如果没有提供默认值,则移除补丁
// - 如果有默认值,则设置为空的 Optional
map.remove(DataComponents.BASE_COLOR);
危险:原型和补丁映射都是PatchedDataComponentMap
哈希码的一部分。因此,映射中的任何组件值都应被视为不可变。在修改数据组件的值后,始终调用#set
或下面讨论的相关方法。
组件持有者(Component Holder)
所有可以持有数据组件的实例都实现了DataComponentHolder
。DataComponentHolder
实际上是DataComponentMap
中只读方法的委托。
// 对于某个 ItemStack stack
// 委托给 'DataComponentMap#get'
@Nullable
DyeColor color = stack.get(DataComponents.BASE_COLOR);
可变的组件持有者(MutableDataComponentHolder)
MutableDataComponentHolder
是 NeoForge 提供的接口,用于支持对组件映射的写操作。Vanilla 和 NeoForge 中的所有实现都使用PatchedDataComponentMap
存储数据组件,因此#set
和#remove
方法也有同名的委托。
此外,MutableDataComponentHolder
还提供了#update
方法,该方法处理获取组件值或提供的默认值(如果未设置),对值进行操作,然后将其设置回映射。操作符可以是UnaryOperator
(接受组件值并返回组件值)或BiFunction
(接受组件值和另一个对象并返回组件值)。
// 对于某个 ItemStack stack
FireworkExplosion explosion = stack.get(DataComponents.FIREWORK_EXPLOSION);
// 修改组件值
explosion = explosion.withFadeColors(new IntArrayList(new int[] {1, 2, 3}));
// 由于我们修改了组件值,应在之后调用 'set'
stack.set(DataComponents.FIREWORK_EXPLOSION, explosion);
// 更新组件值(内部调用 'set')
stack.update(
DataComponents.FIREWORK_EXPLOSION,
// 如果没有组件值,则使用默认值
FireworkExplosion.DEFAULT,
// 返回一个新的 FireworkExplosion 进行设置
explosion -> explosion.withFadeColors(new IntArrayList(new int[] {4, 5, 6}))
);
stack.update(
DataComponents.FIREWORK_EXPLOSION,
// 如果没有组件值,则使用默认值
FireworkExplosion.DEFAULT,
// 提供给函数的对象
new IntArrayList(new int[] {7, 8, 9}),
// 返回一个新的 FireworkExplosion 进行设置
FireworkExplosion::withFadeColors
);
向物品添加默认数据组件
虽然数据组件存储在ItemStack
上,但可以在Item
上设置默认组件映射,以便在构造ItemStack
时作为原型传递。可以通过Item.Properties#component
将组件添加到Item
。
// 对于某个 DeferredRegister.Items REGISTRAR
public static final Item COMPONENT_EXAMPLE = REGISTRAR.register("component",
// 使用 register 而不是其他重载,因为 DataComponentType 尚未注册
() -> new Item(
new Item.Properties()
.component(BASIC_EXAMPLE.value(), new ExampleRecord(24, true))
)
);
如果数据组件应添加到属于 Vanilla 或其他模组的现有物品中,则应在模组事件总线上监听ModifyDefaultComponentEvent
。该事件提供了modify
和modifyMatching
方法,允许修改关联物品的DataComponentPatch.Builder
。构建器可以#set
组件或#remove
现有组件。
// 在模组事件总线上监听
@SubscribeEvent
public void modifyComponents(ModifyDefaultComponentsEvent event) {
// 在西瓜种子上设置组件
event.modify(Items.MELON_SEEDS, builder ->
builder.set(BASIC_EXAMPLE.value(), new ExampleRecord(10, false))
);
// 移除任何具有合成剩余物品的组件的组件
event.modifyMatching(
item -> item.hasCraftingRemainingItem(),
builder -> builder.remove(DataComponents.BUCKET_ENTITY_DATA)
);
}
使用自定义组件持有者
要创建自定义数据组件持有者,持有者对象只需实现MutableDataComponentHolder
并实现缺失的方法。持有者对象必须包含一个表示PatchedDataComponentMap
的字段来实现相关方法。
public class ExampleHolder implements MutableDataComponentHolder {
private int data;
private final PatchedDataComponentMap components;
// 可以提供重载以提供映射本身
public ExampleHolder() {
this.data = 0;
this.components = new PatchedDataComponentMap(DataComponentMap.EMPTY);
}
@Override
public DataComponentMap getComponents() {
return this.components;
}
@Nullable
@Override
public <T> T set(DataComponentType<? super T> componentType, @Nullable T value) {
return this.components.set(componentType, value);
}
@Nullable
@Override
public <T> T remove(DataComponentType<? extends T> componentType) {
return this.components.remove(componentType);
}
@Override
public void applyComponents(DataComponentPatch patch) {
this.components.applyPatch(patch);
}
@Override
public void applyComponents(DataComponentMap components) {
this.components.setAll(p_330402_);
}
// 其他方法
}
DataComponentPatch 和编解码器
为了将组件持久化到磁盘或通过网络发送信息,持有者可以发送整个DataComponentMap
。然而,这通常是一种信息浪费,因为任何默认值都已经存在于数据发送到的位置。因此,我们使用DataComponentPatch
来发送相关数据。DataComponentPatch
仅包含组件映射的补丁信息,而不包含任何默认值。补丁然后应用于接收者位置的原型。
可以通过#patch
从PatchedDataComponentMap
创建DataComponentPatch
。同样,PatchedDataComponentMap#fromPatch
可以构造一个PatchedDataComponentMap
,给定原型DataComponentMap
和DataComponentPatch
。
public class ExampleHolder implements MutableDataComponentHolder {
public static final Codec<ExampleHolder> CODEC = RecordCodecBuilder.create(instance ->
instance.group(
Codec.INT.fieldOf("data").forGetter(ExampleHolder::getData),
DataCopmonentPatch.CODEC.optionalFieldOf("components", DataComponentPatch.EMPTY).forGetter(holder -> holder.components.asPatch())
).apply(instance, ExampleHolder::new)
);
public static final StreamCodec<RegistryFriendlyByteBuf, ExampleHolder> STREAM_CODEC = StreamCodec.composite(
ByteBufCodecs.INT, ExampleHolder::getData,
DataComponentPatch.STREAM_CODEC, holder -> holder.components.asPatch(),
ExampleHolder::new
);
// ...
public ExampleHolder(int data, DataComponentPatch patch) {
this.data = data;
this.components = PatchedDataComponentMap.fromPatch(
// 要应用的原型映射
DataComponentMap.EMPTY,
// 相关的补丁
patch
);
}
// ...
}
通过网络同步持有者数据以及读写数据到磁盘必须手动完成。