数据查询与检索是Hibernate的一个亮点。Hibernate的数据查询方式最主要有3种,它们是:
Hibernate Query Language(HQL) Query By Criteria Native SQL
下面对这3种查询方式分别进行讲解。
Hibernate Query Language(HQL)
Hibernate Query Language(HQL)提供了十分强大的功能,推荐大家使用这种查询方式。HQL具有与SQL语言类似的语法规范,只不过SQL针对表中字段进行查询,而HQL针对持久化对象,它用来取得对象,而不进行update、delete和insert等操作。而且HQL是完全面向对象的,具备继承、多态和关联等特性。
1.from子句
from字句是最简单的HQL语句,例如 from Student,也可以写成select s from Student s。它简单地返回Student类的所有实例。
除了Java类和属性的名称外,HQL语句对大小写并不敏感,所以在上一句HQL语句中,from与FROM是相同的,但是Student与student就不同了,所以上述语句写成from student就会报错。下列程序演示如何通过执行from语句取得所有的Student对象。
Query query = session.createQuery(“from Student”); List list = query.list(); for (int i=0;i<list.size(); i++) { Student stu = (Student)list.get(i); System.out.println(stu.getName()); } |
如果执行HQL语句“from Student, Course”,并不简单地返回两个对象,而是返回这两个对象的的笛卡儿积,这类似于SQL语句中字段的全外连接。在实际应用中,像“from Student, Course”这样的语句几乎不会出现。
2.select子句
有时并不需要得到对象的所有属性,这时可以使用select子句进行属性查询,例如,select s.name from Student s。下面程序演示如何执行这个语句:
Query query = session.createQuery(“select s.name from Student s”); List list = query.list(); for (int i=0;i<list.size(); i++) { String name = (String)list.get(i); System.out.println(ame()); } |
如果要查询两个以上的属性,查询结果会以数组的方式返回,如下所示:
Query query = session.createQuery(“select s.name, s.sex from Student as s”); List list = query.list(); for (int i=0;i<list.size(); i++) { Object obj[] = (Object[])list.get(i); System.out.println(ame(obj[0] + “的性别是:” +obj[1])); } |
在使用属性查询时,由于使用对象数组,操作和理解都不太方便,如果将一个object[]中所有成员封装成一个对象就方便多了。下面的程序将查询结果进行了实例化:
Query query = session.createQuery(“select new Student(s.name, s.sex) from Student s”); List list = query.list(); for (int i=0;i<list.size(); i++) { Student stu = (Student)list.get(i); System.out.println(stu.getName()); } |
要正确运行以上程序,还需要在Student类中加入一个如下的构造函数:
public Student(String name, String sex) { this.name = name; this.sex = sex; } |
3.统计函数查询
可以在HQL中使用函数,经常使用的函数有:
count():统计记录条数 min():求最小值 max():求最大值 sum():求和 age():求平均值
例如,要取得Student实例的数量,可以编写如下HQL语句:
select count(*) from Student |
取得Student的平均年龄的HQL语句如下:
select avg(s.age) from Student as s |
可以使用distinct去除重复数据:
select distinct s.age from Student as s |
4.where子句
HQL也支持子查询,它通过where子句实现这一机制。where子句让用户缩小要返回的实例的列表范围,例如下面语句会返回所有名字为“Bill”的Student实例:
Query query = session.createQuery("from Student as s where s.name='Bill' "); |
where子句允许出现的表达式包括了SQL中可以使用的大多数情况: 数学操作:+,-,*,/ 真假比较操作:=,>=,<=,<>,!=,like 逻辑操作:and,or, not 字符串连接:|| SQL标量函数:例如upper()和lower()
如果子查询返回多条记录,可以用以下的关键字来量化:
all:表示所有的记录。 any:表示所有记录中的任意一条。 some:与any用法相同。 in:与any等价。 exists:表示子查询至少要返回一条记录。
例如,下面语句返回所有学生的年龄都大于22的班级对象:
from Group g where 22<all (select s.age from g.students s) |
下述语句返回在所有学生中有一个学生的年龄等于22的班级:
from Group g where 22=any (select s.age from g.students s) |
或者
from Group g where 22=some (select s.age from g.students s) |
或者
from Group g where |
5.order by 子句
查询返回的列表可以按照任何返回的类或者组件的属性排序:
from Student s order by s.name asc |
asc和desc是可选的,分别代表升序或者降序。
6.连接查询
与SQL查询一样, HQL也支持连接查询,如内连接、外连接和交叉连接。
inner join: 内连接 left outer join:左外连接 right outer join:右外连接 full join: 全连接,但不常用
下面重点讲解内连接查询,左外连接和右外连接查询和内连接大同小异,而全连接几乎不怎么使用。
inner join可以简写为join,例如在查询得到Group对象时,内连接取得对应的Student对象,实现的程序如下。
……//打开Session,开启事务 Student stu = null; //声明Student实例 Group group = null; //声明Group实例 Query query = session.createQuery("from Group g join g.students"); List list = query.list(); Object obj[] = null; //声明对象数组 for(int i=0;i<list.size();i++) { obj = (Object[])list.get(i); //取得集合中的第i个数组 group = (Group)obj[0]; //group是数组中第一个对象 stu = (Student)obj[1]; //stu是数组中第二个对象 System.out.println(stu.getName() + "属于:" +group.getName() ); } ……//提交事务,关闭Session |
Query By Criteria(QBC)
当查询数据时,人们往往需要设置查询条件。在SQL或HQL语句中,查询条件常常放在where子句中。此外,Hibernate还支持Criteria查询(Criteria Query),这种查询方式把查询条件封装为一个Criteria对象。在实际应用中,使用Session的createCriteria()方法构建一个org.hibernate.Criteria实例,然后把具体的查询条件通过Criteria的add()方法加入到Criteria实例中。这样,程序员可以不使用SQL甚至HQL的情况下进行数据查询,如例程1-1所示。
例程1-1 Criteria应用实例
------------------------------------------------------------------------------------------ Criteria cr = session.createCriteria(Student.class); //生成一个Criteria对象 cr.add(Restrictions.eq("name", "Bill"));//等价于where name=’Bill’ List list = cr.list(); Student stu = (Student)list.get(0); System.out.println(stu.getName()); |
1.常用的查询限制方法
在例程1-1中,Restrictions.eq()方法表示equal,即等于的情况。Restrictions类提供了查询限制机制。它提供了许多方法,以实现查询限制。这些方法及其他一些criteria常用查询限制方法列于表1-1中。
表1-1 Criteria Query常用的查询限制方法
方 法 |
说 明 |
||
Restrictions.eq() |
equal,= |
||
Restrictions.allEq() |
参数为Map对象,使用key/value进行多个等于的对比,相当于多个Restrictions.eq()的效果 |
||
Restrictions.gt() |
greater-than, > |
||
Restrictions.lt() |
less-than, < |
||
Restrictions.le() |
less-equal, <= |
||
Restrictions.between() |
对应SQL的between子句 |
||
Restrictions.like() |
对应SQL的like子句 |
||
Restrictions.in() |
对应SQL的in子句 |
||
Restrictions.and() |
|