gpt4 book ai didi

java - 奇怪的 JPA 行为,初始化字段为空

转载 作者:太空狗 更新时间:2023-10-29 22:39:52 25 4
gpt4 key购买 nike

我正在观察实体类的一个非常奇怪的行为,并使用 JPA (hibernate entitymanager 3.3.1.ga) 加载此类的对象。该类有一个(嵌入式)字段,在声明中初始化。该字段的 setter 实现空检查(即当设置空值时会抛出异常)。

...
@Entity
public class Participant extends BaseEntity implements Comparable<Participant> {
...
@Embedded
private AmsData amsData = new AmsData();

public void setAmsData(AmsData amsData) {
Checks.verifyArgNotNull(amsData, "amsdata");
this.amsData = amsData;
}
...
}

当我使用 JPA 获取此对象时,如果数据库中没有嵌入式对象中指定字段的数据,则该字段为空。

...
public class ParticipantJpaDao implements ParticipantDao {
@PersistenceContext
private EntityManager em;

@Override
public Participant getParticipant(Long id) {
return em.find(Participant.class, id);
}
...
}

我用字段上的观察点调试了进程(当字段被访问或修改时应该停止),当字段初始化时我看到一个修改,但是当我从 find 调用中得到结果时,字段是空。

谁能解释一下,为什么会这样?我如何确保该字段不为空,即使在数据库中没有嵌入式对象字段的数据时(除了在查找调用后手动设置它之外)。

最佳答案

JPA 规范没有明确说明如何处理表示可嵌入对象的一组全为空的列。它可以表示一个空引用,或者一个包含所有空字段的对象实例。在这种情况下,Hibernate 选择空引用,但其他 JPA 实现可能会选择后者。

你的 setter 从未被调用的原因是因为 Hibernate 正在通过反射访问你的字段,绕过你实现的 setter。这样做是因为您使用基于字段的访问而不是基于属性的访问。

Chad 的回答将提供您正在寻找的功能,但有一个警告(见下文)。

"...The persistent state of an entity is accessed by the persistence provider runtime[1] either via JavaBeans style property accessors or via instance variables. A single access type (field or property access) applies to an entity hierarchy. When annotations are used, the placement of the mapping annotations on either the persistent fields or persistent properties of the entity class specifies the access type as being either field- or property-based access respectively..." [ejb3 persistence spec]

因此,通过将注释向下移动到 setter ,您告诉 JPA 您想要使用基于属性的访问而不是基于字段的访问。但是,您应该知道,基于字段的访问(正如您当前实现的那样)优于基于属性的访问。不鼓励基于属性的访问有几个原因,但其中一个原因是您被迫为所有持久实体字段添加 getter 和 setter,但您可能不希望这些相同的字段容易受到外部客户端突变的影响。换句话说,使用 JPA 的基于属性的访问会迫使您削弱实体的封装。

关于java - 奇怪的 JPA 行为,初始化字段为空,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/2076644/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com