1

我看了类似的问题,但答案对我没有用..

问题:当我在查询中选择多个父实体时,该查询已连接到其某些@OneToOne子实体,查询会select为其每个子实体执行语句。没有join fetch它按预期执行。


Suser实体有关系;

@OneToOne(optional = false, fetch = FetchType.LAZY)
@Cascade(value = { CascadeType.PERSIST, CascadeType.DELETE })
public SuserStats stats;

@OneToOne(fetch=FetchType.LAZY)
@Cascade(value = {CascadeType.PERSIST})
public Profile profile;

SuserStats与其所有者的实体关系;

@OneToOne
@JoinColumn
public Suser owner;

Profile与其所有者的实体关系;

@OneToOne(optional = false, mappedBy = "profile")
public Suser user;

当我执行查询时;

select u from Suser u where u.id in 
  (select f.targetUser.id from Friendship f where f.sourceUser = ?)

正如预期的那样,在单个查询中仅获取关系的 id。没关系。lazy fetch正在工作中..

但是,当我将其更改为;(这次我要他们拿来)

select u from Suser u left join fetch u.stats s left join fetch u.profile where u.id in 
  (select f.targetUser.id from Friendship f where f.sourceUser = ?)

正在执行 2n+1 个查询。每个查询用于配置文件和统计信息的关系。我尝试了很多这里解释的方法,但它们并没有解决我的问题。我想我在定义父子关系时遗漏了一些基本的东西..

提前致谢..

4

1 回答 1

2

我找到了解决方案;

注意: CascadeType修改与问题和解决方案无关。

注意2:我对实体应用了相同的策略Profile,所以不需要再写了

SuserStats实体;

@MapsId
@OneToOne
@JoinColumn(name="id")
public Suser owner;

Suser实体;

@OneToOne(mappedBy="owner",optional=false,fetch=FetchType.LAZY,cascade=javax.persistence.CascadeType.PERSIST,javax.persistence.CascadeType.REMOVE})
@PrimaryKeyJoinColumn
public SuserStats stats;

通过这些更改,Stats的主列变成了idof Suser(作为外键)。所以默认的惰性关系和按需加入获取正在工作。

于 2013-05-30T00:31:30.827 回答