我有一个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。另一种选择是在控制器和存储库之间创建一个中间服务层,该中间服务层可以公开初始化惰性集合的方法。
Hibernate.initialize()
@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子句来急切地在一次往返数据库中加载角色关联,因此将减轻上述解决方案中两个不同查询引起的性能损失。