4

我很难理解延迟获取,因为当我在一本书中读到它时我不工作,他们说在延迟获取中,jpa 只会在通过 geters 访问实体时加载实体,所以我创建了一个 Arquillian 项目来测试这个概念但它不起作用。这是我的两个实体

package com.actionbazaar.model;

@Entity
@TableGenerator(
        initialValue = 5,
        name = "PERSON_SEQ",
        table = "PERSON_SEQ_TABLE",
        pkColumnName = "SEQ_NAME",
        pkColumnValue = "PERSON",
        valueColumnName = "SEQ_VALUE")
public class Person implements Serializable {

    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    private int id;

    private String fname;
    private String lname;
    @OneToMany(fetch = FetchType.LAZY, mappedBy = "owner", cascade = CascadeType.PERSIST)
    List<Address> addresses;
    //getters and setters 
}

地址

    @Entity
public class Address implements Serializable {

    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    private int id;
    private String city;
    private String zip;
    private String street;

    @ManyToOne
    private Person owner;
//getters and setters
}

我有一个使用这种方法的无状态会话 bean

    public Person getFirstPerson() {
    Person p = em.find(Person.class, 1);
    em.detach(p);
    //why this call does not create an exception
    p.getAddresses().get(0);
    return p;
}

因为我在访问地址之前分离了实体,所以地址列表应该是空的,当我分离它时,它不再由实体管理器管理,所以我不应该为这个人获取地址问题是我可以获取那个人的地址人,即使我有地址字段的惰性获取并在访问地址字段之前分离实体!!!!请一些人解释一下。

另一个测试

Person p= myStatlessSessionBean.getFirstPerson();
myOtherStalessSesionBean.moveAllPeopleToCity("NY");
if(p.getAddresses().get(0).getCity().equals("NY"))
{
  system.out.prinln("person moved");
}
else {
system.out.prinln("person did not move");
} //prompts person did not move
4

2 回答 2

5

是的,伙计,你是对的。你在这里没有做错任何事。我刚刚打开了Pro JPA 2, 2nd edition Book并发现了这个:

这是快照

您正在使用glassfish-embedded,这实际上是导致问题的原因。你的代码没有问题。正如上述书的作者所说,

一些供应商可能会尝试解决这种关系,而其他供应商可能会简单地抛出异常或使属性未初始化。

因此,在您的情况下,关系已解决,而不是延迟加载。只需使用其他供应商实施相同的示例,您就不会遇到任何问题。这里使用 glassfish-embedded,lazyfetch不起作用。否则应该抛出异常,因为变量p是分离的。

这是我还阅读了这条美丽信息的链接

这是上面链接的快照快照

于 2017-02-19T15:21:03.363 回答
1

您只是分离父实体Person。您没有分离子实体地址,当您获得地址时,它指的是仍由持久性上下文管理的实体。

如果您也希望子项也被分离,则应使用 CascadeType.DETACH。

您可能会说,“但是我的 FetchType 设置为 LAZY!”。仅仅因为它是 LAZY,并不意味着该对象为空。Hibernate 为集合类型返回 Proxy 对象,一旦您尝试访问它们,它将填充它们的值。

于 2017-02-19T15:14:05.863 回答