【发布时间】:2019-06-06 18:30:45
【问题描述】:
假设我们有一个简单的实体“客户”,它与实体“地址”具有 OneToOne 关系。外键在地址端。
@Entity
public class Customer extends EntityBase {
@Column(name = "name", nullable = true)
private String name;
@OneToOne(mappedBy = "customer")
private Address address;
// getter, setter, ...
}
@Entity
public class Address extends EntityBase {
@OneToOne(optional = false)
private Customer customer;
@Column(nullable = true)
private String street;
@Column(nullable = true)
private String zip;
@Column(nullable = true)
private String city;
// getter, setter, ...
}
如果您现在使用 hibernate 加载所有 Customer 实体并将结果查询打印到控制台,您可以看到 Hibernate 在内部只触发一个查询。
session.createCriteria(Customer.class).list();
Hibernate 的作用:
select
this_.id as id1_1_1_,
this_.name as name2_1_1_,
address2_.id as id1_0_0_,
address2_.city as city2_0_0_,
address2_.customer_id as customer5_0_0_,
address2_.street as street3_0_0_,
address2_.zip as zip4_0_0_
from
Customer this_
left outer join
Address address2_
on this_.id=address2_.customer_id
如果您使用 QueryDSL 加载 Customer 实体,它将运行一个计数查询(预期和好的),一个针对 Customer 实体的选择查询和一个针对结果集中每个客户的查询。这意味着,如果我想加载 1000 个客户,它将运行 1002 个 SQL 查询。这会造成大量网络流量并降低应用程序的速度。
new HibernateQuery<Customer>(session).from(QCustomer.customer).fetchResults();
Hibernate 与 QueryDSL 的作用:
select
count(customer0_.id) as col_0_0_
from
Customer customer0_
select
customer0_.id as id1_1_,
customer0_.name as name2_1_
from
Customer customer0_
select
address0_.id as id1_0_1_,
address0_.city as city2_0_1_,
address0_.customer_id as customer5_0_1_,
address0_.street as street3_0_1_,
address0_.zip as zip4_0_1_,
customer1_.id as id1_1_0_,
customer1_.name as name2_1_0_
from
Address address0_
inner join
Customer customer1_
on address0_.customer_id=customer1_.id
where
address0_.customer_id=?
问题:
是否可以为 QueryDSL 查询设置全局 FetchMode 之类的东西。在 Hibernate 中,您可以使用 @Fetch(FetchMode.JOIN) 指定它,但不幸的是,QueryDSL 会忽略它。
所以我的目标是使用 QueryDSL 加载 1000 个客户,并且只运行 2 个查询(计数 + 选择)。
我已经知道有一种方法可以指定如下内容:
new HibernateQuery<Customer>(session)
.from(QCustomer.customer)
.leftJoin(QCustomer.customer.address).fetchJoin()
.fetchResults();
但这很容易出错,因为您必须在每个查询中指定它,而且我不想自己声明每个连接。 QueryDSL 在使用谓词时已经自动完成了:
new HibernateQuery<Customer>(session)
.from(QCustomer.customer)
.where(QCustomer.customer.address.street.in("Musterstraße 12"))
.fetchResults();
所以我想使用上面的表达式来加载我的客户,但我不想向我的数据库发起数千个请求,我也不想自己声明每个加入。这可能吗?
我在这里推送了一个示例项目:https://github.com/MatWein/testproject
【问题讨论】: