JPA 查询性能优化 ------EntityGraph篇

当使用JPA进行数据访问时,可能会遇到N+1问题。N+1问题是指在查询关联实体时,JPA会执行额外的N+1次查询,其中N是关联实体的数量。 解决方案:使用JPA2.1特性 @NamedEntityGraph注解,这是JPA推出专门优化解决JPA效率的注解。不管是lazy还是eager,在读取数据的时候,都会有N+1问题。而EntityGraph则直接在查询语句的时候,直接用到用到Left Join,优化了数据库的性能。@NamedEntityGraph注解可以应用于实体类的属性上,用于定义关联实体的抓取策略。通过在查询中使用这个注解,可以一次性获取关联实体的数据,避免额外的查询。

以下是一个使用@NamedEntityGraph注解解决N+1问题的示例代码: 我们先定义实体类Customer,Order,其中Customer和Order之间是一对多的关系。

  • 定义实体类

@NamedEntityGraph(
        name = "customer-with-orders",
        attributeNodes =  {
                @NamedAttributeNode("orders")
        })
@Entity
public class Customer { 
    @Id
    private Integer customerNumber;
    
    @OneToMany(mappedBy = "customer", fetch = FetchType.LAZY)
    private Set<Order> orders;
}

 在上述代码中,@EntityGraph注解指定了要使用的命名实体图"customer-with-orders",指定加载关联实体orders。这样就可以在查询客户列表时一次性获取关联的订单信息,避免N+1问题。也可以配置视图的子视图,如下所示,Order类中关联了Orderdetail实体。

@Entity
public class Order {
        @OneToMany(fetch = FetchType.LAZY, mappedBy = "order")
    private Set<OrderDetail> orderDetail;
}
@NamedEntityGraph(
        name = "customer-with-orders-and-details",
        attributeNodes = {
                @NamedAttributeNode(value = "orders", subgraph = "order-details")
        },
        subgraphs = {@NamedSubgraph(
                name = "order-details",
                attributeNodes = {
                        @NamedAttributeNode(value = "orderDetail")
                }
        )}
)
public class Customer { 
    @Id
    private Integer customerNumber;
    
    @OneToMany(mappedBy = "customer", fetch = FetchType.LAZY)
    private Set<Order> orders;
}

上述代码中,@NamedAttributeNode指定要加载的关联属性orders,和orders的关联实体orderDetail。查询时便可将orders属性和orders属性关联的orderDetail一并查出。

  • 使用视图

方法一:在接口中的方法上加@EntityGraph注解

@Repository
public interface CustomerRepository extends JpaRepository<Customer, Integer> ,BaseRepository<Customer,Integer>{
    @EntityGraph(value = "customer-with-orders",type = EntityGraph.EntityGraphType.FETCH)
    Customer findByCustomerName(String customerName);
}

value属性指定使用的视图名称,对应实体类Customer上@NamedEntityGraph注解中的name属性,EntityGraph.EntityGraphType.FETCH是一个枚举类型,用于指定实体图的加载方式。FETCH表示使用立即加载的方式加载实体图。在使用此方式加载实体图时,相关的属性将会立即从数据库中加载,并与查询结果一同返回,LOAD表示使用延迟加载的方式获取实体及其关联的实体。

 public static enum EntityGraphType {
        LOAD("jakarta.persistence.loadgraph"),
        FETCH("jakarta.persistence.fetchgraph");

        private final String key;

        private EntityGraphType(String value) {
            this.key = value;
        }

        public String getKey() {
            return this.key;
        }
    }
}

方法二:自定义实现类使用entityManager.getEntityGraph(graphName)方法动态获取视图。  

@Repository
public class CustomerRepositoryImpl implements BaseRepository<Customer, Integer> {
    
    @PersistenceContext
    private EntityManager entityManager;
    
    @Override
    public Customer findWithGraph(Integer id, String graphName) {

        EntityGraph entityGraph = entityManager.getEntityGraph(graphName);
        Map<String, Object> properties = new HashMap<>();
        properties.put("javax.persistence.fetchgraph", entityGraph);
        return entityManager.find(Customer.class, id, properties);
    }
}

entityManager.getEntityGraph是一个用于获取实体图的方法。在给定一个实体类型和一个图名称的情况下,它返回与该图关联的EntityGraph对象,该方法所需的参数就是我们定义的视图名称。此方法可将视图名称作为参数传入,动态获取视图,较为灵活。其中properties中的key对应第一种使用方法中的枚举值EntityGraph.EntityGraphType.FETCH。

  • 方法调用

 

public CustomerWithOrdersDto getCustomerAndOrders(Integer id,String graphName) {
    Customer customer = customerRepository.findWithGraph(id, graphName);
    CustomerWithOrdersDto customerWithOrders = customerMapper.customerToCustomerWithOrdersDto(customer);

    List<OrderDto> orders = customer.getOrders().stream()
            .map(order -> orderMapper.OrderToOrderDto(order))
            .collect(Collectors.toList());
    customerWithOrders.setOrders(orders);
    return customerWithOrders;
}

将主键(id)和视图名称(graphName)作为参数传入刚才定义的方法之中,实现一次性加载出视图中配置的所需节点数据。

其中customerMapper.customerToCustomerWithOrdersDto(Customer customer)是一个将Customer类型转换为CustomerWithOrdersDto类型的bean拷贝工具,需要注意的是拷贝过程中可能会触发已查询出来的实体中的懒加载。所以做如下配置忽略orders的拷贝以防止触发懒加载。

@Mapper(componentModel = "spring")
public interface CustomerMapper {
    
    @Mapping(target = "orders",ignore = true)
    CustomerWithOrdersDto customerToCustomerWithOrdersDto(Customer customer);
}

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
【2021年,将Spring全家桶系列课程进行Review,修复顺序等错误。进入2022年,将Spring的课程进行整理,整理为案例精讲的系列课程,并新增高级的Spring Security等内容,通过手把手一步步教你从零开始学会应用Spring,课件将逐步进行上传,敬请期待】 本课程是Spring案例精讲课程的第五部分Spring Data,Spring案例精讲课程以真实场景、项目实战为导向,循序渐进,深入浅出的讲解Java网络编程,助力您在技术工作中更进一步。 本课程聚焦Spring Data的核心知识点:Spring Data Repository、Spring Data JPA(增删改查案例、实体自动生成数据库表、增加新的Repository方法、分页、排序、@NamedQuery、@Query及其分页和排序及参数设置、@NamedEntityGragh实现多对多映射、及QueryHints等)、Spring Data JDBC(增删改查案例、@Query等)的案例介绍, 快速掌握Spring Data的核心知识,快速上手,为学习及工作做好充足的准备。 由于本课程聚焦于案例,即直接上手操作,对于Spring的原理等不会做过多介绍,希望了解原理等内容的需要通过其他视频或者书籍去了解,建议按照该案例课程一步步做下来,之后再去进一步回顾原理,这样能够促进大家对原理有更好的理解。 【通过Spring全家桶,我们保证你能收获到以下几点】 1、掌握Spring全家桶主要部分的开发、实现2、可以使用Spring MVC、Spring Boot、Spring Cloud及Spring Data进行大部分的Spring开发3、初步了解使用微服务、了解使用Spring进行微服务的设计实现4、奠定扎实的Spring技术,具备了一定的独立开发的能力  【实力讲师】 毕业于清华大学软件学院软件工程专业,曾在Accenture、IBM等知名外企任管理及架构职位,近15年的JavaEE经验,近8年的Spring经验,一直致力于架构、设计、开发及管理工作,在电商、零售、制造业等有丰富的项目实施经验  【本课程适用人群】如果你是一定不要错过!  适合于有JavaEE基础的,如:JSP、JSTL、Java基础等的学习者没有基础的学习者跟着课程可以学习,但是需要补充相关基础知识后,才能很好的参与到相关的工作中。 【Spring全家桶课程共包含如下几门】5. 进阶:SpringData 
hibernate-jpa-2.1-api 1.0.2是一个Java持久化规范的实现库。它是基于JPAJava Persistence API)2.1规范的Hibernate实现。Hibernate是一个流行的ORM(对象关系映射)框架,用于在Java应用程序和关系数据库之间进行数据持久化。 该版本的hibernate-jpa-2.1-api是对JPA 2.1规范的实现,并且是Hibernate团队为了确保应用程序与Java EE 7兼容性而发布的一个版本。 JPA是一种使用对象模型操作数据库的标准规范,它提供了一组API,使开发人员可以使用面向对象的方式访问和操作数据库。Hibernate作为一个JPA的实现,提供了许多附加的功能和特性,使得开发人员可以更加简化和灵活地进行数据库操作。 通过使用hibernate-jpa-2.1-api,开发人员可以使用JPA的标准API,以及Hibernate提供的独有特性,来实现应用程序的数据持久化需求。它提供了实体管理器,用于管理实体对象的生命周期,以及CRUD操作。此外,它还提供了用于查询和各种持久化注解的支持。 通常情况下,使用hibernate-jpa-2.1-api需要将其添加到项目的依赖中,并与其他必需的Hibernate库一起使用。开发人员需要熟悉JPA的基本概念和API,并且理解Hibernate特有的扩展和配置选项。 总的来说,hibernate-jpa-2.1-api 1.0.2提供了开发人员在使用JPA进行数据持久化时的基本工具和功能。它是Hibernate团队为了支持JPA 2.1规范而发布的一个版本,开发人员可以使用它来简化和灵活地操作数据库。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值