当前位置: 首页 > 面试题库 >

如何在控制器中从Hibernate / JPA加载延迟获取的项目

鄂琛
2023-03-14
问题内容

我有一个Person类:

@Entity
public class Person {

    @Id
    @GeneratedValue
    private Long id;

    @ManyToMany(fetch = FetchType.LAZY)
    private List<Role> roles;
    // etc
}

与多对多的关系是懒惰的。

在我的控制器中

@Controller
@RequestMapping("/person")
public class PersonController {
    @Autowired
    PersonRepository personRepository;

    @RequestMapping("/get")
    public @ResponseBody Person getPerson() {
        Person person = personRepository.findOne(1L);
        return person;
    }
}

而PersonRepository只是此代码.

public interface PersonRepository extends JpaRepository<Person, Long> {
}

但是,在此控制器中,我实际上需要惰性数据。如何触发加载?

尝试访问它将会失败

无法延迟初始化角色集合:no.dusken.momus.model.Person.roles,无法初始化代理-没有会话

或其他例外情况,具体取决于我的尝试。

我的xml-description,如果需要的话。

谢谢。


问题答案:

你必须对惰性集合进行显式调用才能对其进行初始化(通常的做法是.size()为此目的而调用)。在Hibernate中,有一个专用于此(Hibernate.initialize())的方法,但是JPA没有与此等效的方法。当然,当会话仍然可用时,你必须确保调用已完成,因此请使用注释控制器方法@Transactional。另一种选择是在控制器和存储库之间创建一个中间服务层,该中间服务层可以公开初始化惰性集合的方法。

更新:
请注意,上述解决方案很简单,但是会导致对数据库进行两个不同的查询(一个查询给用户,另一个查询给它的角色)。如果要获得更好的性能,请在Spring Data JPA存储库接口中添加以下方法:

public interface PersonRepository extends JpaRepository<Person, Long> {

    @Query("SELECT p FROM Person p JOIN FETCH p.roles WHERE p.id = (:id)")
    public Person findByIdAndFetchRolesEagerly(@Param("id") Long id);

}

此方法将使用JPQL的fetch join子句来急切地在一次往返数据库中加载角色关联,因此将减轻上述解决方案中两个不同查询引起的性能损失。



 类似资料:
  • 问题内容: 我在JPA实体中的延迟加载属性有问题。我读过许多类似的问题,但它们与spring或hibernate有关,并且他们的后代不适用或没有帮助。 该应用程序是在Wildfly应用程序服务器上运行的JEE和JPA2.1。有两个实体,DAO会话bean和servlet将它们放在一起: 当我运行此代码时,它失败并显示: 我对WebLogic / JPA1使用了非常相似的模式,并且运行平稳。任何的想

  • 问题内容: 我以这样一种方式进行了JPA设置:如果我不使用延迟加载,则几乎将加载整个数据库。我也直接在模型上使用序列化,因此有时我需要初始化代理。 我只想在集合上使用延迟加载。急切地获取一些奇异实体的事实就很好了。但是,无论我如何尝试设置集合,我都永远不会得到代理集合,而我总是会得到完全加载的集合。 这是一些示例代码: 所以这里的问题是,当我检查调试器时,答复的persistantBag-list

  • 我需要从Rest控件获取一个实体及其集合,但我对spring data JPA有一个非常奇怪的行为。 我对带有懒惰加载的实体有一种莫名其妙的关系。

  • 问题内容: 我以这样一种方式进行了JPA设置:如果我不使用延迟加载,则几乎将加载整个数据库。我也直接在模型上使用序列化,因此有时我需要初始化代理。 我只想在集合上使用延迟加载。急切地获取某些奇异实体的事实就很好了。但是,无论我如何尝试设置集合,我都永远不会得到代理集合,而我总是会得到完全加载的集合。 这是一些示例代码: 所以这里的问题是,当我检查调试器时,答复的persistantBag-list

  • 我想避免急切的抓取。如何实现这段代码,这样我就可以随机获取User,它是UserGroup的一部分,它是DocumentType对象中SubmissionUserGroups的一部分。

  • 看了几个网页后,我仍然不明白为什么我的@ElementCollection会急切地加载。我的理解是,默认情况下任何@ElementCollection都应该总是懒洋洋地加载。正如您所看到的,情况并非如此--您将在本文末尾看到控制台的输出。我试图使用@ElementCollection(fetch=fetchType.lazy)注释强制执行惰性加载,但这并没有改变任何事情。有没有人可以解释一下,也可