我很难Brand
从给定Country
的 HQL 中选择一个。我做了一些逆向工程,所以你可以想象我的数据库是如何建模的:
要选择Brand
从右边Country
我不得不经历Address
以及他们City
和State
。
就带注释的映射而言,我已经这样做了:
牌
@Table(name = "brand")
public class Brand extends BasicModel {
private String name;
@OneToMany(targetEntity = Address.class, cascade = CascadeType.ALL,
fetch = FetchType.LAZY, orphanRemoval = true)
@JoinTable(name="brand_address",
joinColumns = { @JoinColumn(name = "brand_id") },
inverseJoinColumns = { @JoinColumn(name = "address_id") })
private Set<Address> address;
地址
@Table(name = "address")
public class Address extends BasicModel {
@NotNull
@ManyToOne
private AddressCity city;
AddressCity
映射到AddressState
which 以AddressCountry
与AddressCity
上面映射的相同方式映射到Address
。
该表brand_address
由 Hibernate 自动创建。
现在使用 SQL 我需要加入所有这些表才能到达,Country
但由于它已与 Hibernate 注释映射,我应该如何编写 HQL 以Brand
从给定的Country
.
信息:当我创建一个包含一组地址的国家/地区时,表格brand_address
将正确填充。
到目前为止有效的只是选择Brand
并且 Hibernate 会将所有填充的对象还给我。我可以通过以下方式做到这一点:
Query query = session
.createQuery("from Brand brand " +
"where brand.name = :brandName " +
"and brand.deleted is null");
query.setParameter("brandName", brandName);
有了这个,我可以轻松地Brands
从其他名称中过滤掉具有该名称的Countries
,但它闻起来不像最佳实践......
当我尝试时,到目前为止还没有奏效:
Query query = session
.createQuery("from Brand brand " +
"where brand.name = :brandName " +
"and Address.city.state.country.code = :countryCode " +
"and brand.deleted is null");
query.setParameter("brandName", brandName);
query.setParameter("countryCode", countryCode);
和
Query query = session
.createQuery("from Brand brand " +
"where brand.name = :brandName " +
"and AddressCountry.code = :countryCode " +
"and brand.deleted is null");
query.setParameter("brandName", brandName);
query.setParameter("countryCode", countryCode);
我该如何解决?有没有办法让我country
不用在上述查询中编写多个连接?我是否应该按照它已经工作的方式全部选择它并将结果过滤回我的 DAO 中?
我一直在互联网上搜索这个,但只找到了关于如何映射和插入信息的信息,而不是关于如何编写将通过表进行选择的查询的信息。
我预先感谢你们美丽的人可以以我的方式发送的任何帮助和建议!
更新 1:我试过添加
left join brand.address as a
with a.city.state.country.code = :countryCode
它导致了一个新的错误:
java.sql.SQLSyntaxErrorException:“on 子句”中的未知列“addresscit3_.state_id”
使用以下日志:
14:02:43.646 [main] DEBUG org.hibernate.hql.internal.ast.util.JoinProcessor - Using FROM fragment [brand brand0_]
14:02:43.646 [main] DEBUG org.hibernate.hql.internal.ast.util.JoinProcessor - Using FROM fragment [inner join address_state addresssta4_ on addresscit3_.state_id=addresssta4_.id]
14:02:43.646 [main] DEBUG org.hibernate.hql.internal.ast.util.JoinProcessor - Using FROM fragment [inner join address_country addresscou5_ on addresssta4_.country_id=addresscou5_.id]
14:02:43.647 [main] DEBUG org.hibernate.hql.internal.ast.util.JoinProcessor - Using FROM fragment [inner join brand_address addresses1_ on brand0_.id=addresses1_.brand_id inner join address address2_ on addresses1_.address_id=address2_.id and (addresscou5_.code=?)]
14:02:43.647 [main] DEBUG org.hibernate.hql.internal.ast.util.JoinProcessor - Using FROM fragment [inner join address_city addresscit3_ on address2_.city_id=addresscit3_.id]
14:02:43.647 [main] DEBUG org.hibernate.hql.internal.antlr.HqlSqlBaseWalker - select >> end [level=1, statement=select]
更新 2:解决方案
使用with
@guillaume 建议的键确实以这样一种方式改变了错误,即连接顺序似乎放错了位置……我的一个朋友建议使用,join fetch
因为延迟加载,我调整了查询,直到一切都按如下所示:
.createQuery("from Brand brand " +
"join fetch brand.addresses as a " +
"where brand.name = :brandName " +
"and a.city.state.country.code = :countryCode " +
"and brand.deleted is null");
该条件Address.city.state.country.code = :countryCode
没有意义,因为 Address 实际上是一个Set
地址(将其命名为地址可能会更清楚)。
试试这个:
from Brand brand
left join brand.address as a
with a.city.state.country.code = :countryCode
本文收集自互联网,转载请注明来源。
如有侵权,请联系 [email protected] 删除。
我来说两句