一尘不染

如何在Spring Controller中获取与JPA和Hibernate的FetchType.LAZY关联

hibernate

我有一个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
,如果需要的话。

谢谢。


阅读 308

收藏
2020-06-20

共1个答案

一尘不染

您必须对延迟集合进行显式调用才能对其进行初始化(通常的做法是.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
子句在一次往返数据库中热切地加载角色关联,因此将减轻上述解决方案中两个截然不同的查询引起的性能损失。

2020-06-20