为了账号安全,请及时绑定邮箱和手机立即绑定

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

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

UYOU 2019-11-22 15:25:50
我有一个Person类:@Entitypublic 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,如果需要的话。谢谢。
查看完整描述

3 回答

?
叮当猫咪

TA贡献1776条经验 获得超12个赞

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


查看完整回答
反对 回复 2019-11-22
?
30秒到达战场

TA贡献1828条经验 获得超6个赞

尽管这是一篇老文章,但请考虑使用@NamedEntityGraph(Javax Persistence)和@EntityGraph(Spring Data JPA)。组合有效。



@Entity

@Table(name = "Employee", schema = "dbo", catalog = "ARCHO")

@NamedEntityGraph(name = "employeeAuthorities",

            attributeNodes = @NamedAttributeNode("employeeGroups"))

public class EmployeeEntity implements Serializable, UserDetails {

// your props

}

然后如下所示的春季回购


@RepositoryRestResource(collectionResourceRel = "Employee", path = "Employee")

public interface IEmployeeRepository extends PagingAndSortingRepository<EmployeeEntity, String>           {


    @EntityGraph(value = "employeeAuthorities", type = EntityGraphType.LOAD)

    EmployeeEntity getByUsername(String userName);


}


查看完整回答
反对 回复 2019-11-22
?
波斯汪

TA贡献1811条经验 获得超4个赞

你有一些选择


根据RJ建议,在存储库上编写一个返回初始化实体的方法。

更多工作,最佳性能。


使用OpenEntityManagerInViewFilter可使整个请求的会话保持打开状态。

工作量少,通常在网络环境中可以接受。


需要时,使用帮助器类来初始化实体。

较少的工作,在未选择OEMIV时(例如在Swing应用程序中)很有用,但在存储库实现中一次也可以初始化任何实体可能也有用。


对于最后一个选项,我编写了一个实用程序类JpaUtils来在某些时候初始化实体。


例如:


@Transactional

public class RepositoryHelper {


    @PersistenceContext

    private EntityManager em;


    public void intialize(Object entity, int depth) {

        JpaUtils.initialize(em, entity, depth);

    }

}


查看完整回答
反对 回复 2019-11-22
  • 3 回答
  • 0 关注
  • 629 浏览

添加回答

举报

0/150
提交
取消
意见反馈 帮助中心 APP下载
官方微信