0

我知道冬眠。我有一个sql语句

SELECT VERSION_ID,D_BEGIN,D_END, 
 (SELECT NAME FROM zvit where ZVIT_ID=version.ZVIT_ID) as name  
 FROM version  
 where  VERSION_ID in  
 (SELECT term.VERSION_ID FROM term  
 where term.PERIOD_MONTH= :periodMonth and term.PERIOD_TYPE= :periodType and term.PERIOD_YEAR= :periodYear )  
 order by 1 ;

我尝试使用 createCriteria 和 HQL 来实现它。

    DetachedCriteria subQuery  = DetachedCriteria.forClass(TermData.class)
            .add(Restrictions.eq("periodmonth",  periodMonth))
            .add(Restrictions.eq("periodtype",  periodType))
            .add(Restrictions.eq("periodyear", periodYear))
            .setProjection(Projections.projectionList() 
                    .add(Projections.property("versionId.versionId"))
                    );          
    Criteria versionCriteria = session.createCriteria(VersionData.class)
            .addOrder(Order.asc("versionId"))
            .add(Subqueries.propertyIn("versionId", subQuery))
            .createAlias("zvitId", "zvitId", org.hibernate.sql.JoinType.INNER_JOIN)
            .setProjection(Projections.projectionList() 
                    .add(Projections.property("versionId"))
                    .add(Projections.property("dbegin"))
                    .add(Projections.property("dend"))
                    .add(Projections.property("zvitId.name"), "name")

                    );  

总部:

            Query query = session.createQuery(""
                    + "from VersionData as version "
//                  + "inner join version.zvitId as zvit "
                    + "where version.versionId in "
                    + "(select term.versionId from TermData as term "
                    + "where term.periodmonth= :periodmonth and term.periodtype= :periodtype and term.periodyear= :periodyear)");

问题是这个 HQL 的执行时间要长 10 倍。并做了很多不必要的查询。我尝试用带注释的字符串来做,它有所改进,但仍然比 createCriteria 查询长 5 倍,此外我无法进行此转换

List<VersionData> queryResult = query.list(); 当我得到 VersionData 和 ZvitData 对象时。这似乎不是问题,但我仍然不明白为什么我不只收到 VersionData 对象。但还有一个更重要的问题是有什么方法可以改进这个 HQL 语句,使其与 mysql 或 createCriteria 查询几乎同时执行。

版本数据定义

@Entity
@Table(name = "version")
public class VersionData implements Serializable
{

    /**
     * 
     */
    private static final long serialVersionUID = 7355281418627668744L;

    @Id 
    @Column(name="VERSION_ID")  
    private String versionId;   

    /**
     * user dbegin
     */
    @Column(name = "D_BEGIN")
    @Type(type = "date")
    private Date dbegin;

    /** user dend */
    @Column(name = "D_END")
    @Type(type = "date")
    private Date dend;  


    /**
     * user zvitId
     */
    @ManyToOne
    @JoinColumn(name="ZVIT_ID") 
    private ZvitData zvitId;

    @OneToMany(targetEntity=TermData.class, mappedBy = "versionId", cascade=javax.persistence.CascadeType.ALL, fetch=FetchType.LAZY)
    private List<TermData> terms;
//getters, setters, hashcode, equals
}
4

1 回答 1

1

好吧,使用您发布的条件查询,您只获取您需要的字段,而使用您的 HQL 查询您获取整个 Hibernate 托管实体 (VersionData)。根据此实体的配置方式,您可能会看到触发其他查询以填充此实体的字段。

您也可以使用 HQL 执行与使用标准相同的操作,并且在我看来,它比使用标准更易于理解,因此更易于维护。只需使用适当的构造函数为您希望获取的字段定义一个 java 类 MyClass,然后调用一个类似于以下内容的 HQL 查询:

"select new path.to.myclass.MyClass ( "
                + " alias1.field1, "
                + " alias2.field2, "
                + " ... "

其中 aliases 是您在 HQL 查询中赋予实体的名称(例如 version 是您编写的查询中 VersionData 的别名)。没有投影,没有额外的抽象,只是一个类似 SQL 的查询。

您可以直接获取字段,然后处理返回的结果,而不是定义一个新类来保存从 HQL 查询返回的字段(我觉得这更方便),如下面的链接所示:

如何仅检索 JPQL 或 HQL 中实体的某些字段?JPQL 或 HQL 中的 ResultSet 等价物是什么?

于 2016-03-04T08:47:25.783 回答