Capability 并没有解决数据的持久化的问题。
准确地来说,Capability 系统只有一个 IStorage
可以提供缺省缺省状态下的数据持久化,但它不一定适用于所有的情况。
在 Forge 内部使用的 CapabilityDispatcher
中,它会检查每一个收集到的 ICapabilityProvider
是否还实现了 INbtSerializable
,并在识别到 INbtSerializable
时使用它来序列化对应的 ICapabilityProvider
。但即便如此,这也是 CapabilityDispatcher
自行实现的序列化机制,而非 Capability 系统的特性。
一言以蔽之,通过 AttachCapabilitiesEvent
追加的 ICapabilityProvider
可以通过多实现 INbtSerializable
的方式来保存数据;若不是通过 AttachCapabilitiesEvent
追加到别的地方的 Capability,我们需要自行解决数据保存的问题。
我们可以直接把数据写入 TileEntity
自己的 NBT 中。
@Override
public NBTTagCompound writeToNBT(NBTTagCompound data) {
data.setTag("something", this.capObject.serializeNBT());
data.setInteger("another_thing", this.someEnergy);
return super.writeToNBT(data);
}
@Override
public void readFromNBT(NBTTagCompound data) {
super.readFromNBT(data);
this.capObject.deserializeNBT(data.getCompoundTag("something"));
this.someEnergy = data.getInteger("another_thing");
}
我们可以直接把数据写入 ItemStack
自己的 NBT 中。
@Override
public ICapabilityProvider initCapabilities(ItemStack stack, NBTTagCompound nbt) {
return new ICapabilityProvider() {
// 实现略,基本思路是通过匿名类捕获 stack 对象,
// 借 ItemStack 的 NBTTagCompound 来持久化数据。
};
}
若是通过 AttachCapabilitiesEvent
附加的功能,也可以通过实现 INbtSerializable
的方式保存数据。然而这里有一个坑:
一般来说,ItemStack
的同步其实都是 GUI 在负责的,因为 ItemStack
它本身并不与任何游戏底层机制耦合,和底层耦合的是 GUI 及 TileEntity
。也因为此,同步一个 ItemStack
实际上比表面上的情况还要复杂——
实体死亡时,Capability 数据会丢失,需要手动同步。但实际上,对于 EntityPlayer
,从末地返回时,Capability 数据理应被保存。处于某种原因,这个特性至今仍未被实现。我们需要自行处理这个问题:
@SubscribeEvent
public static void reSyncDataOnPlayerDeath(PlayerEvent.Clone event) {
// PlayerEvent.Clone 这个事件会在两个情况下发布:
// 1. 玩家实体死亡
// 2. 玩家穿越维度,尤其是从末地的传送门返回主世界。
// event.isWasDeath() 可区分这两个情况,但有鉴于上述数据保存的特性未能实现,
// 我们只能无条件转移所有数据。
MyCapability cap = event.getEntityPlayer().getCapability(MyCapability.INSTANCE, null);
if (cap != null) {
MyCapability original = event.getOriginal().getCapability(MyCapability.INSTANCE, null);
// 然后将 original 里的数据逐个同步进 cap 中。
}
}