JPQL:多重连接。如何做我的命名查询?

Posted

技术标签:

【中文标题】JPQL:多重连接。如何做我的命名查询?【英文标题】:JPQL : multiple join. How to do my namedquery? 【发布时间】:2012-08-14 10:23:01 【问题描述】:

这是我的数据库的快速图表。

http://img17.imageshack.us/img17/2474/mpd.png

在这个方案中,我创建了 JPA 实体(对于所有带有红色方块的表)。

我想创建一个 JPQL 查询来获取所有具有引用的飞机,引用由参数给出的引用类型定义。

我试过了: SELECT DISTINCT a FROM Aircraft a JOIN FETCH a.references r WHERE EXISTS (SELECT ref FROM Reference ref WHERE ref = r AND ref.referenceType.id = :id)

但我有一个错误,因为 Eclipse 不喜欢 JOIN FETCH a.references *r* 中的别名,并且请求在 JUnit 测试中不起作用。

这里我的实体没有 getter/setter :

飞机

@Entity
@Table(name = "T_R_AIRCRAFT_AIR", uniqueConstraints = @UniqueConstraint(columnNames = "AIR_NAME"))
public class Aircraft implements java.io.Serializable 

    @Id
    @Column(name = "AIR_ID", unique = true, nullable = false)
    @TableGenerator(name="aircraftSeqStore", 
        table="T_S_APP_SEQ_STORE_AST", 
        pkColumnName="AST_SEQ_NAME",
        valueColumnName = "AST_SEQ_VALUE",
        pkColumnValue = "T_R_AIRCRAFT_AIR.AIR_ID", 
        allocationSize=1)
    @GeneratedValue(strategy=GenerationType.TABLE, 
        generator="aircraftSeqStore")       
    private Integer id;

    @Column(name = "AIR_NAME", unique = true, nullable = false, length = 50)
    private String name;

    @ManyToMany(fetch = FetchType.LAZY)
    @JoinTable(name = "T_J_REF_AIR_RFA", 
               joinColumns =  @JoinColumn(name = "RFA_AIR_ID", nullable = false, updatable = false) , 
               inverseJoinColumns =  @JoinColumn(name = "RFA_REF_ID", nullable = false, updatable = false) )
    private Set<Reference> references = new HashSet<Reference>(0);

    @ManyToMany(fetch = FetchType.LAZY)
    @JoinTable(name = "T_J_MAN_AIR_MNA", 
               joinColumns =  @JoinColumn(name = "MNA_AIR_ID", nullable = false, updatable = false) , 
               inverseJoinColumns =  @JoinColumn(name = "MNA_MAN_ID", nullable = false, updatable = false) )  
    private Set<Manual> manuals = new HashSet<Manual>(0);

    @OneToMany(fetch = FetchType.LAZY, 
               mappedBy = "aircraft",
               cascade =  CascadeType.REMOVE )
    private Set<UserConfig> userConfigs = new HashSet<UserConfig>(0);

** 参考 **

@Entity
@Table(name = "T_E_REFERENCE_REF", 
       uniqueConstraints = @UniqueConstraint(columnNames = "REF_IDENTIFIER"))
public class Reference implements java.io.Serializable 

    @Id
    @Column(name = "REF_ID", unique = true, nullable = false)
    @TableGenerator(name="referenceSeqStore", 
        table="T_S_APP_SEQ_STORE_AST", 
        pkColumnName="AST_SEQ_NAME",
        valueColumnName = "AST_SEQ_VALUE",
        pkColumnValue = "T_E_REFERENCE_REF.REF_ID", 
        allocationSize=1)
    @GeneratedValue(strategy=GenerationType.TABLE, generator="referenceSeqStore")                   
    private Integer id;

    @Column(name = "REF_IDENTIFIER", unique = true, nullable = false, length = 50)  
    private String identifier;

    @Column(name = "REF_LINK")  
    private String link;

    @Column(name = "REF_OBSERVATIONS", length = 4000)
    private String observations;

    @ManyToOne(fetch = FetchType.LAZY)
    @JoinColumn(name = "REF_RFT_ID", nullable = false)  
    private ReferenceType referenceType;

    @ManyToOne(fetch = FetchType.LAZY)
    @JoinColumn(name = "REF_MAN_ID")    
    private Manual manual;  

    @OneToMany(fetch = FetchType.LAZY, 
               mappedBy = "reference",
               cascade =  CascadeType.REMOVE )
    private Set<Translation> translations = new HashSet<Translation>(0);

    @ManyToMany(fetch = FetchType.LAZY)
    @JoinTable(name = "T_J_REF_AIR_RFA", 
               joinColumns =  @JoinColumn(name = "RFA_REF_ID", nullable = false, updatable = false) ,
               inverseJoinColumns =  @JoinColumn(name = "RFA_AIR_ID", nullable = false, updatable = false) )
    private Set<Aircraft> aircrafts = new HashSet<Aircraft>(0);

** 参考类型 **

@Entity
@Table(name = "T_R_REFERENCE_TYPE_RFT", 
       uniqueConstraints = @UniqueConstraint(columnNames = "RFT_TYPE"))
public class ReferenceType implements java.io.Serializable 

    @Id
    @Column(name = "RFT_ID", unique = true, nullable = false)
    @TableGenerator(name="referenceTypeSeqStore", 
        table="T_S_APP_SEQ_STORE_AST", 
        pkColumnName="AST_SEQ_NAME",
        valueColumnName = "AST_SEQ_VALUE",
        pkColumnValue = "T_R_REFERENCE_TYPE_RFT.RFT_ID", 
        allocationSize=1)
    @GeneratedValue(strategy=GenerationType.TABLE, generator="referenceTypeSeqStore")                   
    private Integer id;

    @Column(name = "RFT_TYPE", unique = true, nullable = false, length = 50)    
    private String type;

    @OneToMany(fetch = FetchType.LAZY, 
               mappedBy = "referenceType", 
               cascade =  CascadeType.REMOVE )    
    private Set<Reference> references = new HashSet<Reference>(0);

    @OneToMany(fetch = FetchType.LAZY, 
               mappedBy = "referenceType",
               cascade =  CascadeType.REMOVE )    
    private Set<UserConfig> userConfigs = new HashSet<UserConfig>(0);

PS:我忘了说在飞机和参考之间添加了一个表格。这是手动表。但我认为没有影响。

PS2:JPA 是由 Hibernate 实现的。

知道如何进行多重连接吗? 谢谢!

【问题讨论】:

如何加入一个已经从一个加入的实体?即如何将 A 与 B 与 C 连接起来?似乎很容易将 A 与 B 结合,A 与 C 结合,但对于其他人来说,我没有成功。 【参考方案1】:

您可以更轻松地执行查询,如下所示:

SELECT DISTINCT a FROM Aircraft a JOIN FETCH a.references r 
LEFT JOIN FETCH a.manuals 
WHERE r.referenceType.id = :id

我添加了join fetch a.manuals,否则我得到了LazyInitializationException。新增LEFT,防止缺少手册影响输出。

【讨论】:

最后,我的 JUnit 测试有一个 pb。延迟加载太多,加载了太多集合,我的内存不足。我对 JPQL 有困难。所以我不知道我的 JPQL 是否正常工作。 JPA 验证器似乎不接受在连接提取之后放置别名。暂时你的回答不允许我理解它是如何工作的。这个 JPQL 是否正确:SELECT DISTINCT a FROM Aircraft a JOIN FETCH a.references JOIN FETCH a.references.referenceType WHERE a.references.referenceType.id = :id?感谢您的耐心等待。 不,此查询将不起作用,因为此路径 a.references.referenceType 无效。想想你自己:你引用了references 集合的referenceType 属性,后者显然没有。 我知道使用 join fetch 是为了“加载”为延迟加载配置的集合?那么是否可以创建嵌套查询?在 SQL 中,我们可能首先通过查询来选择 ReferenceType Id 的所有引用,然后通过一些关节来获取飞机(通过连接表)。 在 JPQL 中,您应该忘记连接表,因为在这里您使用 @Entities、@ElementCollections、@Embeddabbles 等。在 JPQL 中,仅当您想通过 @ManyToMany 或 @OneToMany 关联访问实体时才需要加入,就像当前情况一样。否则,如果 Aircraft 与 Reference 存在 @OneToOne 或 @ManyToOne 关联,那么您可以按如下方式制定查询:SELECT DISTINCT a FROM Aircraft a LEFT JOIN FETCH a.manuals WHERE a.reference.referenceType.id = :id 好的。谢谢你的这些解释。所以,如果我总结一下,所有@ManyToMany@OneToMany 关联都需要连接,但@ManyToOne@OneToOne....除非存在延迟加载,我们仍然需要进行连接fetch 以“加载”我们的实体。准确吗?

以上是关于JPQL:多重连接。如何做我的命名查询?的主要内容,如果未能解决你的问题,请参考以下文章

如何在JPQL中创建“findBy Date”命名查询?

如何编写一个 JPQL 查询来查找此连接中未找到的记录?

我可以限制使用 JPQL 命名查询的 Spring Data 查询方法的查询结果吗?

如何在 JPQL 中进行“深度”获取连接?

JPQL 限制查询 [重复]

Hibernate:如何通过连接字符串来创建 JPQL?