快捷搜索:

另类查询 Hibernate HQL 深度历险

传统的SQL说话采纳的是布局化的查询措施,而这种措施对付查询以工具形式存在的数据却力所不及。幸运的是,Hibernate为我们供给了一种语法类似于SQL的说话,Hibernate查询说话(HQL),和SQL不合的是,HQL是一种面向工具的查询说话,它可以查询以工具形式存在的数据。是以,本文就HQL若何事情以及若何应用HQL展开了深入的评论争论。

SQL本身是异常强大年夜的。当SQL的这种强大年夜和处置惩罚面向工具数据的能力相结应时,就孕育发生了HQL。和SQL一样,HQL供给了富厚的查询功能,如投影查询、聚合函数、分组和约束。任何繁杂的SQL都可以映射成HQL。

本文的第一部分将评论争论HQL的简单用法。第二部分将评论争论在HQL中若何根据高低文关系进行查询。在第三部分将以一个例子来阐明若何在实际利用中应用HQL。

进入HQL天下

一个ORM框架是建立在面向工具的根基上的。最好的例子是Hibernate若何供给类SQL查询。虽然HQL的语法类似于SQL,但实际上它的查询目标是工具。HQL拥有面向工具说话的所有的特点,这此中包括多态、承袭和组合。这就相称于一个面向工具的SQL,为了供给更强大年夜的功能,HQL还供给了很多的查询函数。这些函数可以被分为四类:

1.投影函数

2.约束函数

3.聚合函数

4.分组函数

应用HQL可以建立简单的查询,也可以建立更繁杂的查询。在本文中并不评论争论那些异常繁杂的查询,如含有子查询和很多连接的查询。本文只评论争论连接两个表的查询。现在让我们开始靠近HQL吧!

投影

如谓投影,便是一个可以造访的工具或工具的属性。在HQL中,可以应用from和select子句来完成这个事情。

from子句返回指定的类的所有实例。如from Order将返回Order类的所有实例。换句话说,以上的查询相称于以下的SQL语句:

select * from order

from 是最简单的查询子句。from后面可以跟一个或多个类名(类名也可以带有又名)。为了获得Order和Product的所有实例,可以应用如下的查询:

from Order, Product

和类名一样,又名也可以在from后应用,如下代码如示:

from Order as o, Product p

当查询很繁杂时,加入又名可以削减语句的长度。我们可以看看如下的SQL语句:

select o.*, p.* from order o, product p where o.order_id = p.order_id

我们可以很轻易看出,上面的查询是一对多的关系。在HQL中相称于一个类中包孕多个其它类的实例。是以,以上的SQL写成HQL便是:

from Order as o inner join o.products as product

现在让我们斟酌别的一个从表中获得指定属性的环境。这便是最常用的select子句。这在HQL中的事情要领和SQL中一样。而在HQL中,假如只是想获得类的属性的话,select语句是着末的选择。以上的SQL可以应用select子句改成如下的HQL语句:

select product from Order as o inner join o.products as product

以上的HQL语句将返回Order中的所有Products实例。假如要获得工具的某一个属性,可以将HQL语句写成如下的形式:

select product.name from Order as o inner join o.products as product

假如要获得多个工具的属性,可以将HQL语句写成如下形式:

select o.id, product.name from Order as o inner join o.products as product

接下来,我们将进入下一个议题。假设我们必要根据某些前提获得数据。那么以上所述的HQL语句将无法满意需求。为了达到这一目的,我们就要用到下面将要评论争论的约束子句。

约束

从以上可知,投影返回的是所有的数据。但在大年夜多半时刻我们并不必要这么多半据。这就必要对数据进行过滤。在HQL中过滤数据的子句和SQL一样,也是where。它的语法类似于SQL,经由过程where子句,可以对行进行过滤。我们可以看看下面的SQL语句:

select * from orders where id = ‘1234’

这条查询语句返回了id即是1234的所有的字段。和这条SQL对等的是下面的HQL语句:

select o from Order o where o.id=’1234’

从以上两条语句可以看出,它们的where子句异常相似。而它们独一的不合是SQL操作的是记录,而HQL操作的是工具。在HQL中,除了where子句可以过滤数据外,having子句也可以做到这一点(关于having子句的具体内容我将在分组部分评论争论)。投影和约束是两个基础的操作,这两个操作再加上聚合函数的话,那HQL将变得加倍强大年夜。下面我们就来评论争论什么是聚合。

聚合

上述的查询都是将每一个记录(工具)当做一个单位,而假如应用聚合,可以将一类记录(工具)当做一个单位。然后再对每一类的记录(工具)进行一系列地操作,如对某一列取匀称值、乞降、统计行数等等。HQL支持以下的聚合函数:

1.avg(…), sum(…)

2.min(…), max(…)

3.count(*), count(…), count(distinct…), count(all…)

以上的聚合函数都返回数值类型。这些操作都可以在select子句中应用,如下所示:

select max(o.priceTotal) + max(p.price) from Order o join o.products p group by o.id

以上的HQL语句返回了两个值的和:orders表中的priceTotal的最大年夜值和products表中的price的最大年夜值之和。我们还可以应用having子句对分组进行过滤。如我们想按id统计priceTotal小于1000的数量可按如下的HQL语句去实现:

select count(o) from Order o having o.priceTotal

我们还可以将聚合函数和having子句一路应用。如我们要按products表的id统计price小于amount的匀称数的产品数量,HQL语句如下:

select count(p) from Product p having p.price

从上面的一系列的HQL语句可以看出,所有经由过程SQL实现的,都可以经由过程HQL来实现。

分组

在上一部分,已经涉及到了分组的观点。分组操作的是行的聚拢。它根据某一列(属性)对记录集进行分组。这统统是经由过程group子句实现的。如下的例子描述了group子句的一样平常用法。

select count(o) from Order o having o.priceTotal >= 1200 and o.priceTotal

HQL中的分组和SQL中的分组类似。总之,除了一些对SQL的特殊扩展外,其它所有的SQL功能都可以应用HQL描述。在接下来的部分,让我们举例阐明若何在Java中应用HQL。

在Java中应用HQL

到现在为止,我们已经进修了HQL的基础用法。接下来我们举一个例子来阐明若何在Java中应用HQL。下面的例子只给出了主要的部分,因为本文只是评论争论HQL的用法,是以,关于Hibernate的一些设置和在main()函数中调用Hibernate的部分并未给出,读者可以参考相关的文当。现在让我们看看下面的例子。

下面是必须引用的包

import java.util.List;

import org.hibernate.*;

import org.hibernate.cfg.*

import com.Order;

下面是类的声明

public class MyOrder

{

… …

}

下面让我们来实现MyOrder类的构造函数

public class MyOrder

{

SessionFactory sf;

public MyOrder()

{

Configuration cfg = new Configuration().addClass(Order.class);

sf = cfg.buildSessionFactory();

}

… …

}

下面的getOrder函数根据priceTotal的区间值返回Order工具。

public class MyOrder

{

…. ….

public Order getOrder(String lower, String upper)

{

// 打开一个会话

Session sess = sf.openSession();

// HQL语句

String query = "select o from o "

+ "Order as o join o.products as p "

+ "where o.priceTotal > :priceTotalLower"

+ "and o.priceTotal

下面的main函数将测试MyOrder类

public class MyOrder

{

… …

public static void main(String args[])

{

Order o=MyOrder().getOrder(“100”, “300”);

System.out.println(“id=”+ o.id);

… …

}

}

小结

上述的代码演示了若何在Java中应用HQL,但HQL还有两点必要留意一下:

1.HQL并不区分字母的大年夜小写,但在HQL中的Java类和属性名必须和实际的类和属性名同等。如SELECT和select之间可以交换,但Order和order却代表不合的含义。

2.假如HQL中引用的类未被导入,在HQL中必须引用详细的包。如本例中,假如com.Order未被导入,在HQL中必须将Order写成com.Order。

您可能还会对下面的文章感兴趣: