问题描述
我希望有人以前可能遇到过这个或类似的问题,可以帮助我:)
我有一个 KryoNet 服务器/客户端架构,用于发送消息。 这些消息之一包含类“WorldEntity”的实例。 这个实体看起来像这样:
@Entity
@Inheritance(strategy = InheritanceType.TABLE_PER_CLASS)
public abstract class WorldEntity {
@OnetoMany
private Collection<EntityComponent> components;
}
我正在使用 spring + hibernate 将此实体保存到我的 postgresql 数据库中。 这一切正常。
当我现在将这个实体发送给客户端时,kryonet 序列化程序尝试延迟加载组件,这是不应该的,因为客户端上没有休眠或数据库。 实际上,所有数据都已经包含在实体中。
我在某处读到,可以创建自定义序列化对象并将其添加到 KryoNet 客户端,以禁用此休眠加载:
Kryo kryoSerializer = new Kryo() {
@Override
public Serializer<?> getDefaultSerializer(final Class type) {
if (AbstractPersistentCollection.class.isAssignableFrom(type)) {
return new FieldSerializer(kryoSerializer,type);
}
return super.getDefaultSerializer(type);
}
};
不幸的是,这不能在 Kryo 客户端的构造函数中用作序列化对象。
任何帮助将不胜感激!
亲切的问候 达斯汀
解决方法
我建议您开始考虑为此目的添加 DTO,以避免为了安全和网络性能问题而向您的客户暴露过多信息。我认为这是 Blaze-Persistence Entity Views 的完美用例。
我创建了该库以允许在 JPA 模型和自定义接口或抽象类定义的模型之间轻松映射,例如类固醇上的 Spring Data Projections。这个想法是,您可以按照自己喜欢的方式定义目标结构(域模型),并通过 JPQL 表达式将属性(getter)映射到实体模型。
您的用例的 DTO 模型可能如下所示,其中包含 Blaze-Persistence Entity-Views:
@EntityView(WorldEntity.class)
public interface WorldEntityDto {
@IdMapping
Long getId();
String getName();
Set<EntityComponentDto> getComponents();
@EntityView(EntityComponent.class)
interface EntityComponentDto {
@IdMapping
Long getId();
String getName();
}
}
查询是将实体视图应用于查询的问题,最简单的就是通过 id 查询。
WorldEntityDto a = entityViewManager.find(entityManager,WorldEntityDto.class,id);
Spring Data 集成允许您像使用 Spring Data Projections 一样使用它:https://persistence.blazebit.com/documentation/entity-view/manual/en_US/index.html#spring-data-features
Page<WorldEntityDto> findAll(Pageable pageable);
最好的部分是,它只会获取实际需要的状态!