【问题标题】:JPA: How to avoid generating multiple "select from" queriesJPA:如何避免生成多个“select from”查询
【发布时间】:2018-04-20 06:23:43
【问题描述】:

这不是经典的 N+1 问题。我的问题是在 Jpa 中使用投影和 DTO 对象。

我有 JPA 查询的下一个方法:

public List<MeterDTO> getAllBrokenMeterByHouseServ(House house, Serv serv, Date dt) {
    Query query =em.createQuery("select new MeterDTO(m, g.kart.lsk, nvl(e.tp,0)) from Meter m "
        + "join m.exs e with m.id=e.meter.id "
        + "join m.meterLog g with m.meterLog.id=g.id "
        + "join g.kart k with g.kart.id=k.id and :dt between k.dt1 and k.dt2 " 
        + "join g.serv s with g.serv.id=s.id "
        + "join k.kw kw with k.kw.id=kw.id "
        + "join kw.house h with kw.house.id=h.id "
        + "where s.id = :servId "
        + "and kw.house.id = :houseId "
        + "and :dt between e.dt1 and e.dt2 and nvl(e.tp,0) in (2,3,4) "
        + "");
    query.setParameter("servId", serv.getId());
    query.setParameter("houseId", house.getId());
    query.setParameter("dt", dt);
    return query.getResultList();
}

我从上面的查询中提取记录到 数据传输对象:

    meterDao.getAllBrokenMeterByHouseServ(house, serv, dt2).stream().forEach(t-> {
        log.info("meter.id={}, lsk={}, tp={} ", t.getMeter().getId(), t.getLsk(), t.getTp());
    });

MeterDTO:

@Getter @Setter
public class MeterDTO {

    private Meter meter;
    private Integer lsk;
    private Double tp;

    public MeterDTO(Meter meter, Integer lsk, Double tp) {
        super();
        this.meter = meter;
        this.lsk = lsk;
        this.tp = tp;
    }
}

为什么hibernate会产生一个主查询:

select
    meter0_.ID as col_0_0_,
    kart3_.lsk as col_1_0_,
    nvl(exs1_.TP,
    0) as col_2_0_ 
from
    MT.METER meter0_ 
inner join
    MT.METER_EXS exs1_ 
        on meter0_.ID=exs1_.FK_METER 
        and (
            meter0_.ID=exs1_.FK_METER
        ) 
inner join
    MT.METER_LOG meterlog2_ 
        on meter0_.FK_METER_LOG=meterlog2_.ID 
        and (
            meter0_.FK_METER_LOG=meterlog2_.ID
        ) 
inner join
    AR.KART kart3_ 
        on meterlog2_.FK_KLSK_OBJ=kart3_.FK_KLSK_OBJ 
        and (
            kart3_.lsk=kart3_.lsk 
            and (
                ? between kart3_.DT1 and kart3_.DT2
            )
        ) 
inner join
    AR.KW kw6_ 
        on kart3_.FK_KW=kw6_.ID 
        and (
            kart3_.FK_KW=kw6_.ID
        ) 
inner join
    AR.HOUSE house7_ 
        on kw6_.FK_HOUSE=house7_.ID 
        and (
            kw6_.FK_HOUSE=house7_.ID
        ) 
inner join
    TR.SERV serv5_ 
        on meterlog2_.FK_SERV=serv5_.ID 
        and (
            meterlog2_.FK_SERV=serv5_.ID
        ) 
where
    serv5_.ID=? 
    and kw6_.FK_HOUSE=? 
    and (
        ? between exs1_.DT1 and exs1_.DT2
    ) 
    and (
        nvl(exs1_.TP, 0) in (
            2 , 3 , 4
        )
    )

以及具有不同绑定参数“?”的多个查询加载每个实体:

select
    meter0_.ID as ID1_44_0_,
    meter0_.FK_K_LSK as FK_K_LSK2_44_0_,
    meter0_.FK_METER_LOG as FK_METER_LOG4_44_0_,
    meter0_.TRANS_RATIO as TRANS_RATIO3_44_0_ 
from
    MT.METER meter0_ 
where
    meter0_.ID=?

如何避免这个问题?我想在一个主查询中加载所有实体 Meter。 有可能吗?

我用:

<spring-framework.version>5.0.5.RELEASE</spring-framework.version>
<hibernate.version>5.1.0.Final</hibernate.version>

任何帮助将不胜感激。

更新1

我将我的 JPA 查询代码简化为:

    public List<MeterDTO> getAllBrokenMeterByHouseServ(House house, Serv serv, Date dt) {
        Query query =em.createQuery("select new com.ric.bill.dto.MeterDTO(m) from Meter m ");
}

但它仍然会产生多个查询:

    select
        meter0_.ID as ID1_44_0_,
        meter0_.FK_K_LSK as FK_K_LSK2_44_0_,
        meter0_.FK_METER_LOG as FK_METER_LOG4_44_0_,
        meter0_.TRANS_RATIO as TRANS_RATIO3_44_0_ 
    from
        MT.METER meter0_ 
    where
        meter0_.ID=?
20-04-2018 12:52:49.482 [main] DEBUG o.h.l.p.e.p.i.ResultSetProcessorImpl - Starting ResultSet row #0
20-04-2018 12:52:49.482 [main] DEBUG org.hibernate.SQL - 
    select
        meter0_.ID as ID1_44_0_,
        meter0_.FK_K_LSK as FK_K_LSK2_44_0_,
        meter0_.FK_METER_LOG as FK_METER_LOG4_44_0_,
        meter0_.TRANS_RATIO as TRANS_RATIO3_44_0_ 
    from
        MT.METER meter0_ 
    where
        meter0_.ID=?
<Skipped>

很奇怪!

upd2仪表实体:

@SuppressWarnings("serial")
@Entity
@Table(name = "METER", schema="MT")
@Getter @Setter
public class Meter extends Base implements java.io.Serializable, Storable {

    public Meter (){

    }

    @Id
    @GeneratedValue(strategy = GenerationType.AUTO)
    @Column(name = "ID", updatable = false, nullable = false)
    protected Integer id;

    @ManyToOne(fetch = FetchType.LAZY)
    @JoinColumn(name="FK_METER_LOG", referencedColumnName="ID")
    private MeterLog meterLog ; 

    @OneToMany(fetch = FetchType.LAZY, cascade = CascadeType.ALL, orphanRemoval=true)
    @JoinColumn(name="FK_METER", referencedColumnName="ID")
    @BatchSize(size = 50)
    private List<Vol> vol = new ArrayList<Vol>(0);

    @OneToMany(fetch = FetchType.LAZY)
    @JoinColumn(name="FK_METER", referencedColumnName="ID")
    @BatchSize(size = 50)
    private List<MeterExs> exs = new ArrayList<MeterExs>(0);

    @Column(name = "TRANS_RATIO", updatable = true, nullable = true)
    private Double trRatio; 

}

【问题讨论】:

  • 使用JOIN FETCH 而不是JOIN
  • 如果您不想获取相关数据,请在 OneToMany 注释中使用属性 FetchType.LAZY
  • @Alberto 他想获取相关数据(从查询中可以清楚地看到),但他希望它们在单个查询中。他正在根据结果创建一个投影。
  • 能把Meter实体的代码放上来吗?
  • 请查看我的更新 upd1 和 upd2

标签: java jpa jpql


【解决方案1】:

在 DTO 中,您有“Meter Meter”字段,在 Meter 字段中,您有“MeterLog Meterlog”等。在这种情况下,Hibernate 还为完整对象的字段加载。这个 DTO 非常复杂。尝试创建更扁平的对象:

public class MeterDTO {

    private Integer meterId
    private Double meterTrRatio
    private Integer lsk;
    private Double tp;
    (...)

查询将是:

(...) new MeterDTO(m.id, m.trans_ratio, g.kart.lsk (...)

之后,您可以将 DTO 扩展到您想要的下一个字段。

【讨论】:

    【解决方案2】:

    接受的答案建议更改 DTO,这并不总是一个可接受的解决方案。

    这是一个无需更改 DTO 的解决方案。

    这样写你的 HQL:

    from Meter m 
    join m.exs e with m.id=e.meter.id 
    join m.meterLog g with m.meterLog.id=g.id 
    join g.kart k with g.kart.id=k.id and :dt between k.dt1 and k.dt2 " 
    join g.serv s with g.serv.id=s.id "
    join k.kw kw with k.kw.id=kw.id "
    join kw.house h with kw.house.id=h.id "
    (more joins and wheres)
    

    请注意,不应有任何select

    getResultList 会给你List&lt;Object[]&gt;。每个条目都是 {Meter, m.exs, m.meterLog, g.kart, ....} 的数组。选择您需要的并设置您的MeterDTO

    就我而言:

    jpa 回购

        @Query("from Bind bind "
            + "left join Employee employee "
            + "with bind.empCode = employee.empCode "
            + "where bind.accountName = :hiveAccount and bind.disabled = 1 ")
        List<Object[]> listMembers(@Param("hiveAccount") String hiveAccount);
    

    DTO

    public class BindDTO {
        Bind bind;
        Employee emp;
    
        public BindDTO(Object[] objs) {
            this((Bind) objs[0], (Employee) objs[1]);
        }
    

    服务

    myRepo.listMembers(hiveAccount).stream().map(BindDTO::new).collect(Collectors.toList());
    

    【讨论】:

      猜你喜欢
      • 2011-12-15
      • 2016-05-15
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2015-02-16
      相关资源
      最近更新 更多