序列化数据

虽然 ImmutableDataManipulator 存储数据是十分美妙的,然而对于重启服务器这一操作,数据的存储就束手无策了。然而,每一个 DataManipulator 都实现了 DataSerializable 接口,这意味着我们就可以使用 DataContainer 序列化(Serialize)数据,同时使用 DataBuilder 反序列化(Deserialize)数据。

在通过初步地从特定的 DataManipulator 转换为较为普遍的结构之后, DataContainer 还可以起进行进一步的转换。

DataContainer 和 DataView

作为一个可以表示任何数据的通用结构, DataView 支持多个值,甚至包括嵌套的 DataView 作为值,这使得其看起来像一个树形的结构。每一个索引都使用 DataQuery 去表示。 DataContainerDataView 的根。

每个 DataSerializable 都提供一个名为 toContainer() 的方法将其转换为 DataContainer 。作为示例,通过对一个 HealthData 的实例的 toContainer() 方法调用将会返回一个包含有两个数据的 DataContainer ,一个表示当前生命值,一个表示最大生命值。每个 DataQuery 都通过对应的 Key 确定。

import org.spongepowered.api.data.DataContainer;
import org.spongepowered.api.data.key.Keys;

DataContainer serializedHealth = healthData.toContainer();
double currentHealth = serializedHealth.getDouble(Keys.HEALTH.getQuery()).get();
currentHealth == healthData.health().get();  // true

现在就是 DataBuilder 大显身手的时候了,它可以把一个 DataContainer 反序列化成一个 HealthData 的实例。 DataManager 负责注册和管理这些事情。它的获取方式包括一个 Game 的实例,或者 Sponge 类等。 DataManager 提供了一个方法用于获取合适的 DataBuilder 用于反序列化指定的 Class 和 额外的 DataBuilder 从而一步完成反序列化操作。下面的代码示例是等价的。

代码示例:较长代码的反序列化操作

import org.spongepowered.api.data.DataView;
import org.spongepowered.api.data.manipulator.mutable.entity.HealthData;
import org.spongepowered.api.util.persistence.DataBuilder;

import java.util.Optional;

public Optional<HealthData> deserializeHealth(DataView container) {
    final Optional<DataBuilder<HealthData>> builder = Sponge.getDataManager().getBuilder(HealthData.class);
    if (builder.isPresent()) {
        return builder.get().build(container);
    }
    return Optional.empty();
}

代码示例:较短代码的反序列化操作

import org.spongepowered.api.data.manipulator.mutable.entity.HealthData;

public Optional<HealthData> deserializeHealth(DataView container) {
    return Sponge.getDataManager().deserialize(HealthData.class, container);
}

如果没有用于 HealthDataDataBuilder 被注册,或者指定的 DataContainer 为空,方法 deserializeHealth 将会返回一个 Optional.empty() 。如果 DataContainer 中含有非法数据,那么将抛出一个 InvalidDataException 异常。

DataTranslator

在 Sponge 中,一般的实现有 MemoryDataViewMemoryDataContainer ,它们只存储在内存中,因此在服务器重新启动后便不复存在。为了持久存储一个 DataContainer ,我们首先必须将其转换为可存储形式。

通过使用 DataTranslators#CONFIGURATION_NODE 实现 DataTranslator ,我们可以将一个 DataView 转换为一个 ConfigurationNode ,反之亦然。 ConfigurationNode 可以使用 Configurate Library 来写入和读取可被持久保存的文件。

代码示例:把 HealthData 序列化为配置文件数据

import ninja.leaping.configurate.ConfigurationNode;
import org.spongepowered.api.data.persistence.DataTranslator;
import org.spongepowered.api.data.persistence.DataTranslators;

public ConfigurationNode translateToConfig(HealthData data) {
    final DataTranslator<ConfigurationNode> translator = DataTranslators.CONFIGURATION_NODE;
    final DataView container = data.toContainer();
    return translator.translate(container);
}

代码示例:把配置文件数据反序列化为 HealthData

import java.util.Optional;

public Optional<HealthData> translateFromConfig(ConfigurationNode node) {
    final DataTranslator<ConfigurationNode> translator = DataTranslators.CONFIGURATION_NODE;
    final DataView container = translator.translate(node);
    return deserializeHealth(container);
}