Hibernate数据查询
2012-02-02 15:24
225 查看
来自:《J2EE开源编程精要15讲:整合Eclipse、Struts、Hibernate和Spring的JavaWeb开发》
9.1Hibernate数据查询
讲要点:
Hibernate数据查询
利用关联关系操纵对象
Hibernate事务
Hibernate的Cache管理
9.1Hibernate数据查询
数据查询与检索是Hibernate的一个亮点。Hibernate的数据查询方式主要有3种,它们是:
HibernateQueryLanguage(HQL)
CriteriaQuery
NativeSQL
下面对这3种查询方式分别进行讲解。
9.1.1HibernateQueryLanguage(HQL)
HibernateQueryLanguage(HQL)提供了十分强大的功能,推荐大家使用这种查询方式。HQL具有与SQL语言类似的语法规范,只不过SQL针对表中字段进行查询,而HQL针对持久化对象,它用来取得对象,而不进行update、delete和insert等操作。而且HQL是完全面向对象的,具备继承、多态和关联等特性。
1.from子句
from字句是最简单的HQL语句,例如fromStudent,也可以写成selectsfromStudents。它简单地返回Student类的所有实例。
除了Java类和属性的名称外,HQL语句对大小写并不敏感,所以在上一句HQL语句中,from与FROM是相同的,但是Student与student就不同了,所以上述语句写成fromstudent就会报错。下列程序演示如何通过执行from语句取得所有的Student对象。
Queryquery=session.createQuery(“fromStudent”);
Listlist=query.list();
for(inti=0;i<list.size();i++)
{
Studentstu=(Student)list.get(i);
System.out.println(stu.getName());
}
如果执行HQL语句“fromStudent,Course”,并不简单地返回两个对象,而是返回这两个对象的的笛卡儿积,这类似于SQL语句中字段的全外连接。在实际应用中,像“fromStudent,Course”这样的语句几乎不会出现。
2.select子句
有时并不需要得到对象的所有属性,这时可以使用select子句进行属性查询,例如,selects.namefromStudents。下面程序演示如何执行这个语句:
Queryquery=session.createQuery(“selects.namefromStudents”);
Listlist=query.list();
for(inti=0;i<list.size();i++){
Stringname=(String)list.get(i);
System.out.println(name());
}
如果要查询两个以上的属性,查询结果会以数组的方式返回,如下所示:
Queryquery=session.createQuery(“selects.name,s.sexfromStudentass”);
Listlist=query.list();
for(inti=0;i<list.size();i++){
Objectobj[]=(Object[])list.get(i);
System.out.println(name(obj[0]+“的性别是:”+obj[1]));
}
在使用属性查询时,由于使用对象数组,操作和理解都不太方便,如果将一个object[]中所有成员封装成一个对象就方便多了。下面的程序将查询结果进行了实例化:
Queryquery=session.createQuery(“selectnewStudent(s.name,s.sex)fromStudents”);
Listlist=query.list();
for(inti=0;i<list.size();i++){
Studentstu=(Student)list.get(i);
System.out.println(stu.getName());
}
要正确运行以上程序,还需要在Student类中加入一个如下的构造函数:
publicStudent(Stringname,Stringsex)
{
this.name=name;
this.sex=sex;
}
3.统计函数查询
可以在HQL中使用函数,经常使用的函数有:
count():统计记录条数
min():求最小值
max():求最大值
sum():求和
age():求平均值
例如,要取得Student实例的数量,可以编写如下HQL语句:
selectcount(*)fromStudent
取得Student的平均年龄的HQL语句如下:
selectavg(s.age)fromStudentass
可以使用distinct去除重复数据:
selectdistincts.agefromStudentass
4.where子句
HQL也支持子查询,它通过where子句实现这一机制。where子句让用户缩小要返回的实例的列表范围,例如下面语句会返回所有名字为“Bill”的Student实例:
Queryquery=session.createQuery("fromStudentasswheres.name='Bill'");
where子句允许出现的表达式包括了SQL中可以使用的大多数情况:
数学操作:+,-,*,/
真假比较操作:=,>=,<=,<>,!=,like
逻辑操作:and,or,not
字符串连接:||
SQL标量函数:例如upper()和lower()
如果子查询返回多条记录,可以用以下的关键字来量化:
all:表示所有的记录。
any:表示所有记录中的任意一条。
some:与any用法相同。
in:与any等价。
exists:表示子查询至少要返回一条记录。
例如,下面语句返回所有学生的年龄都大于22的班级对象:
fromGroupgwhere22<all(selects.agefromg.studentss)
下述语句返回在所有学生中有一个学生的年龄等于22的班级:
fromGroupgwhere22=any(selects.agefromg.studentss)
或者
fromGroupgwhere22=some(selects.agefromg.studentss)
或者
fromGroupgwhere22in(selects.agefromg.studentss)
5.orderby子句
查询返回的列表可以按照任何返回的类或者组件的属性排序:
fromStudentsorderbys.nameasc
asc和desc是可选的,分别代表升序或者降序。
6.连接查询
与SQL查询一样,HQL也支持连接查询,如内连接、外连接和交叉连接。
innerjoin:内连接
leftouterjoin:左外连接
rightouterjoin:右外连接
fulljoin:全连接,但不常用
下面重点讲解内连接查询,左外连接和右外连接查询和内连接大同小异,而全连接几乎不怎么使用。
innerjoin可以简写为join,例如在查询得到Group对象时,内连接取得对应的Student对象,实现的程序如下。
……//打开Session,开启事务
Studentstu=null;//声明Student实例
Groupgroup=null;//声明Group实例
Queryquery=session.createQuery("fromGroupgjoing.students");
Listlist=query.list();
Objectobj[]=null;//声明对象数组
for(inti=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
9.1.2CriteriaQuery方式
当查询数据时,人们往往需要设置查询条件。在SQL或HQL语句中,查询条件常常放在where子句中。此外,Hibernate还支持Criteria查询(CriteriaQuery),这种查询方式把查询条件封装为一个Criteria对象。在实际应用中,使用Session的createCriteria()方法构建一个org.hibernate.Criteria实例,然后把具体的查询条件通过Criteria的add()方法加入到Criteria实例中。这样,程序员可以不使用SQL甚至HQL的情况下进行数据查询,如例程9-1所示。
例程9-1Criteria应用实例
------------------------------------------------------------------------------------------
Criteriacr=session.createCriteria(Student.class);//生成一个Criteria对象
cr.add(Restrictions.eq("name","Bill"));//等价于wherename=’Bill’
Listlist=cr.list();
Studentstu=(Student)list.get(0);
System.out.println(stu.getName());
1.常用的查询限制方法
在例程9-1中,Restrictions.eq()方法表示equal,即等于的情况。Restrictions类提供了查询限制机制。它提供了许多方法,以实现查询限制。这些方法及其他一些criteria常用查询限制方法列于表9-1中。
表9-1CriteriaQuery常用的查询限制方法
方法说明
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()and关系
Restrictions.or()or关系
Restrictions.isNull()判断属性是否为空,为空返回true,否则返回false
Restrictions.isNotNull()与Restrictions.isNull()相反
Order.asc()根据传入的字段进行升序排序
Order.desc()根据传入的字段进行降序排序
MatchMode.EXACT字符串精确匹配,相当于“like'value'”
MatchMode.ANYWHERE字符串在中间位置,相当于“like'%value%'”
MatchMode.START字符串在最前面的位置,相当于“like'value%'”
MatchMode.END字符串在最后面的位置,相当于“like'%value'”
例1:查询学生名字以t开头的所有Student对象。
Criteriacr=session.createCriteria(Student.class);
cr.add(Restrictions.like(“name”,“t%”))
Listlist=cr.list();
Studentstu=(Student)list.get(0);
或者使用另一种方式:
Criteriacr=session.createCriteria(Student.class);
cr.add(Restrictions.like(“name”,“t”,MatchMode.START))
Listlist=cr.list();
Studentstu=(Student)list.get(0);
例2:查询学生姓名在Bill,Jack和Tom之间的所有Student对象。
String[]names={“Bill”,“Jack”,“Tom”}
Criteriacr=session.createCriteria(Student.class);
cr.add(Restrictions.in(“name”,names))
Listlist=cr.list();
Studentstu=(Student)list.get(0);
例3:查询学生的年龄age等于22或age为空(null)的所有Student对象。
Criteriacr=session.createCriteria(Student.class);
cr.add(Restrictions.eq(“age”,newInteger(22));
cr.add(Restrictions.isNull(“age”));
Listlist=cr.list();
Studentstu=(Student)list.get(0);
例4:查询学生姓名以字母F开头的所有Student对象,并按姓名升序排序。
Criteriacr=session.createCriteria(Student.class);
cr.add(Restrictions.like(“name”,“F%”);
cr.addOrder(Order.asc(“name”));
Listlist=cr.list();
Studentstu=(Student)list.get(0);
调用Order.asc的方法应是Criteria的addOrder()方法。
2.连接限制
在Criteria查询中使用FetchMode来实现连接限制。在HQL语句中,可以通过fetch关键字来表示预先抓取(Eagerfetching),如下所示:
fromGroupg
leftjoinfetchg.studentss
whereg.namelike'%2005'
可以使用Criteria的API完成同样的功能,如下所示:
Criteriacr=session.createCriteria(Group.class);
cr.setFetchMode(“students”,FetchMode.EAGER);
cr.add(Restrictions.like(“name”,“2005”,MatchMode.END))
Listlist=cr.list();
以上两种方式编写的代码,都使用相同的SQL语句完成它们的功能,如下所示:
selectg.*,s.*fromGroupg
leftouterjoinStudents
ong.id=s.group_id
whereg.namelike'%2005'
相关文章推荐
- Hibernate只查父表数据,关闭Hql多态查询
- Hibernate中原生sql查询多个表,数据映射
- hibernate_数据分页, 查询总数
- Hibernate有哪几种查询数据的方式
- Hibernate原生SQL查询数据转换为HQL查询数据方法
- Hibernate的数据过滤查询
- 在hibernate中查询数据的 session中get 和load的区别:
- Hibernate联合主键映射规则和数据查询原理
- Hibernate数据查询 利用关联关系操纵对象 Hibernate事务 Hibernate的Cache管理
- 利用hibernate映射的接口,只重复显示查询到的第一条数据
- 用hibernate翻译的sql在sqlplus可以查询出数据,但是程序查询为空
- Hibernate 查询不到数据
- Hibernate入门05 - 基本数据查询
- Hibernate使用sql查询出map类型数据
- Hibernate查询,数据库中无数据正常,有数据出现java.lang.NullPointerException 空指针异常
- Hibernate数据查询
- hibernate_查询简介: 分页, 数据总数, 结果集的包装, 位置参数的设置
- hibernate的数据查询 4
- Hibernate使用原生SQL适应复杂数据查询
- Hibernate工具类和插入查询数据例子