`
廖世勇
  • 浏览: 100989 次
  • 性别: Icon_minigender_1
  • 来自: 湖南郴州
社区版块
存档分类
最新评论

Hibernate

 
阅读更多

第17章Native SQL查询

你也可以使用你的数据库的Native SQL语言来查询数据。这对你在要使用数据库的某些特性的时候(比如说在查询提示或者Oracle中的 CONNECT关键字),这是非常有用的。这就能够扫清你把原来直接使用SQL/JDBC 的程序迁移到基于 Hibernate应用的道路上的障碍。

Hibernate3允许你使用手写的sql来完成所有的create,update,delete,和load操作(包括存储过程)

17.1.创建一个基于SQL的Query

SQL查询是通过SQLQuery接口来控制的,它是通过调用Session.createSQLQuery()方法来获得

List cats = sess.createSQLQuery("select {cat.*} from cats cat")

.addEntity("cat",Cat.class);

.setMaxResults(50);

.list();

这个查询指定了:

·SQL查询语句,它带一个占位符,可以让Hibernate使用字段的别名.

·查询返回的实体,和它的SQL表的别名.

addEntity()方法将SQL表的别名和实体类联系起来,并且确定查询结果集的形态。

addJoin()方法可以被用于载入其他的实体和集合的关联,TODO:examples!

原生的SQL查询可能返回一个简单的标量值或者一个标量和实体的结合体。

Double max = (Double) sess.createSQLQuery("select max(cat.weight) asmaxWeight from cats cat")

.addScalar("maxWeight", Hibernate.DOUBLE);

.uniqueResult();

17.2.别名和属性引用

上面使用的{cat.*}标记是 "所有属性" 的简写.你可以显式地列出需要的字段,但是你必须让Hibernate 为每一个属性注入字段的别名.这些字段的站位符是以字段别名为前导,再加上属性名.在下面的例子里,我们从一个其他的表(cat_log) 中获取Cat对象,而非Cat对象原本在映射元数据中声明的表.注意我们甚至在where子句中也可以使用属性别名. 对于命名查询,{}语法并不是必需的.你可以在17.3“命名SQL查询”得到更多的细节.

String sql = "select cat.originalId as {cat.id}, " +

"cat.mateid as {cat.mate},cat.sex as {cat.sex}, " +

"cat.weight*10 as{cat.weight}, cat.name as {cat.name} " +

"from cat_log cat where{cat.mate} = :catId"

List loggedCats = sess.createSQLQuery(sql)

.addEntity("cat",Cat.class)

.setLong("catId",catId)

.list();

注意:如果你明确地列出了每个属性,你必须包含这个类和它的子类的属性! and itssubclasses!

17.3.命名SQL查询

可以在映射文档中定义查询的名字,然后就可以象调用一个命名的HQL查询一样直接调用命名SQL查询.在这种情况下,我们 需要调用addEntity()方法.

<sql-query name="mySqlQuery">

<returnalias="person" class="eg.Person"/>

SELECT person.NAME AS{person.name},

person.AGE AS{person.age},

person.SEX AS{person.sex}

FROM PERSON person WHERE person.NAMELIKE 'Hiber%'

</sql-query>

List people = sess.getNamedQuery("mySqlQuery")

.setMaxResults(50)

.list();

一个命名查询可能会返回一个标量值.你必须使用<return-scalar>元素来指定字段的别名和 Hibernate类型

<sql-query name="mySqlQuery">

<return-scalarcolumn="name" type="string"/>

<return-scalarcolumn="age" type="long"/>

SELECT p.NAME AS name,

p.AGE AS age,

FROM PERSON p WHERE p.NAME LIKE'Hiber%'

</sql-query>

<return-join>和<load-collection>元素分别用作 外连接和定义那些初始化集合的查询

17.3.1.使用return-property来明确地指定字段/别名

使用<return-property>你可以明确的告诉Hibernate使用哪些字段,这和使用{}-语法 来让Hibernate注入它自己的别名是相反的.

<sql-query name="mySqlQuery">

<returnalias="person" class="eg.Person">

<return-propertyname="name" column="myName"/>

<return-propertyname="age" column="myAge"/>

<return-propertyname="sex" column="mySex"/>

</return>

SELECT person.NAME AS myName,

person.AGE AS myAge,

person.SEX AS mySex,

FROM PERSON person WHEREperson.NAME LIKE :name

</sql-query>

<return-property>也可用于多个字段,它解决了使用{}-语法不能细粒度控制多个字段的限制

<sql-query name="organizationCurrentEmployments">

<returnalias="emp" class="Employment">

<return-propertyname="salary">

<return-columnname="VALUE"/>

<return-columnname="CURRENCY"/>

</return-property>

<return-propertyname="endDate" column="myEndDate"/>

</return>

SELECT EMPLOYEE AS{emp.employee}, EMPLOYER AS {emp.employer},

STARTDATE AS{emp.startDate}, ENDDATE AS {emp.endDate},

REGIONCODE as{emp.regionCode}, EID AS {emp.id}, VALUE, CURRENCY

FROM EMPLOYMENT

WHERE EMPLOYER = :id ANDENDDATE IS NULL

ORDER BY STARTDATE ASC

</sql-query>

注意在这个例子中,我们使用了<return-property>结合{}的注入语法. 允许用户来选择如何引用字段以及属性.

如果你映射一个识别器(discriminator),你必须使用<return-discriminator>来指定识别器字段

17.3.2.使用存储过程来查询

Hibernate 3引入了对存储过程查询的支持. 存储过程必须返回一个结果集,作为Hibernate能够使用的第一个外部参数. 下面是一个Oracle9和更高版本的存储过程例子.

CREATE OR REPLACE FUNCTION selectAllEmployments

RETURN SYS_REFCURSOR

AS

st_cursor SYS_REFCURSOR;

BEGIN

OPEN st_cursor FOR

SELECT EMPLOYEE, EMPLOYER,

STARTDATE, ENDDATE,

REGIONCODE, EID, VALUE, CURRENCY

FROM EMPLOYMENT;

RETURN st_cursor;

END;

在Hibernate里要要使用这个查询,你需要通过命名查询来映射它.

<sql-query name="selectAllEmployees_SP"callable="true">

<return alias="emp"class="Employment">

<return-propertyname="employee" column="EMPLOYEE"/>

<return-propertyname="employer" column="EMPLOYER"/>

<return-propertyname="startDate" column="STARTDATE"/>

<return-propertyname="endDate" column="ENDDATE"/>

<return-propertyname="regionCode" column="REGIONCODE"/>

<return-propertyname="id" column="EID"/>

<return-propertyname="salary">

<return-columnname="VALUE"/>

<return-columnname="CURRENCY"/>

</return-property>

</return>

{ ? = callselectAllEmployments() }

</sql-query>

注意存储过程当前仅仅返回标量和实体.现在不支持<return-join>和<load-collection>

17.3.2.1.使用存储过程的规则和限制

为了在Hibernate中使用存储过程,你必须遵循一些规则.不遵循这些规则的存储过程将不可用.如果你仍然想要使用他们, 你必须通过session.connection()来执行他们.这些规则针对于不同的数据库.因为数据库 提供商有各种不同的存储过程语法和语义.

对存储过程进行的查询无法使用setFirstResult()/setMaxResults()进行分页。

对于Oracle有如下规则:

·存储过程必须返回一个结果集.它通过返回SYS_REFCURSOR实现(在Oracle9或10),在Oracle里你需要定义一个REF CURSOR 类型

·推荐的格式是 { ? = call procName(<parameters>) } 或 { ? = call procName }(这更像是Oracle规则而不是Hibernate规则)

对于Sybase或者MS SQL server有如下规则:

·存储过程必须返回一个结果集。.注意这些servers可能返回多个结果集以及更新的数目.Hibernate将取出第一条结果集作为它的返回值, 其他将被丢弃。

·如果你能够在存储过程里设定SET NOCOUNT ON,这可能会效率更高,但这不是必需的。

17.4.定制SQL用来create,update和delete

Hibernate3能够使用定制的SQL语句来执行create,update和delete操作。在Hibernate中,持久化的类和集合已经 包含了一套配置期产生的语句(insertsql, deletesql, updatesql等等),这些映射标记<sql-insert>, <sql-delete>, and <sql-update>重载了 这些语句。

<class name="Person">

<id name="id">

<generatorclass="increment"/>

</id>

<propertyname="name" not-null="true"/>

<sql-insert>INSERT INTOPERSON (NAME, ID) VALUES ( UPPER(?), ? )</sql-insert>

<sql-update>UPDATE PERSONSET NAME=UPPER(?) WHERE ID=?</sql-update>

<sql-delete>DELETE FROMPERSON WHERE ID=?</sql-delete>

</class>

这些SQL直接在你的数据库里执行,所以你可以自由的使用你喜欢的任意语法。但如果你使用数据库特定的语法, 这当然会降低你映射的可移植性。

如果设定callable,则能够支持存储过程了。

<class name="Person">

<id name="id">

<generatorclass="increment"/>

</id>

<propertyname="name" not-null="true"/>

<sql-insertcallable="true">{call createPerson (?, ?)}</sql-insert>

<sql-deletecallable="true">{? = call deletePerson (?)}</sql-delete>

<sql-updatecallable="true">{? = call updatePerson (?, ?)}</sql-update>

</class>

参数的位置顺序是非常重要的,他们必须和Hibernate所期待的顺序相同。

你能够通过设定日志调试级别为org.hiberante.persister.entity,来查看Hibernate所期待的顺序。在这个级别下, Hibernate将会打印出create,update和delete实体的静态SQL。如果想看到预想中的顺序。记得不要将定制SQL包含在映射文件里, 因为他们会重载Hibernate生成的静态SQL。

在大多数情况下(最好这么做),存储过程需要返回插入/更新/删除的行数,因为Hibernate对语句的成功执行有些运行时的检查。 Hibernate常会把进行CUD操作的语句的第一个参数注册为一个数值型输出参数。

CREATE OR REPLACE FUNCTION updatePerson (uid IN NUMBER, uname IN VARCHAR2)

RETURN NUMBER IS

BEGIN

update PERSON

set

NAME = uname,

where

ID = uid;

return SQL%ROWCOUNT;

END updatePerson;

17.5.定制装载SQL

你可能需要声明你自己的SQL(或HQL)来装载实体

<sql-query name="person">

<return alias="p"class="Person" lock-mode="upgrade"/>

SELECT NAME AS {p.name}, ID AS{p.id} FROM PERSON WHERE ID=? FOR UPDATE

</sql-query>

这只是一个前面讨论过的命名查询声明,你可以在类映射里引用这个命名查询。

<class name="Person">

<id name="id">

<generatorclass="increment"/>

</id>

<propertyname="name" not-null="true"/>

<loaderquery-ref="person"/>

</class>

这也可以用于存储过程

TODO: 未完成的例子

<sql-query name="organizationEmployments">

<load-collectionalias="empcol" role="Organization.employments"/>

SELECT {empcol.*}

FROM EMPLOYMENT empcol

WHERE EMPLOYER = :id

ORDER BY STARTDATE ASC, EMPLOYEEASC

</sql-query>

<sql-query name="organizationCurrentEmployments">

<return alias="emp"class="Employment"/>

<synchronizetable="EMPLOYMENT"/>

SELECT EMPLOYEE AS{emp.employee}, EMPLOYER AS {emp.employer},

STARTDATE AS{emp.startDate}, ENDDATE AS {emp.endDate},

REGIONCODE as {emp.regionCode}, ID AS{emp.id}

FROM EMPLOYMENT

WHERE EMPLOYER = :id AND ENDDATEIS NULL

ORDER BY STARTDATE ASC

</sql-query>


来自:http://docs.huihoo.com/hibernate/reference-v3_zh-cn/querysql.html

分享到:
评论

相关推荐

    hibernate jar包:hibernate-commons-annotations-4.0.1.Final.jar等

    Hibernate.jar包,Hibernate可以应用在任何使用JDBC的场合,包含 hibernate-commons-annotations-4.0.1.Final.jar hibernate-core-4.1.12.Final.jar hibernate-ehcache-4.1.12.Final.jar hibernate-entitymanager-...

    Hibernate入门到精通

    "Hibernate入门到精通" Hibernate 是一个基于Java的ORM(Object-Relational Mapping,对象关系映射)框架,它提供了一种简洁高效的方式来访问和操作关系数据库。下面是 Hibernate 的主要知识点: Hibernate 简介 ...

    hibernate jar包程序文件

    hibernate-commons-annotations-4.0.1.Final.jar hibernate-core-4.1.12.Final.jar hibernate-ehcache-4.1.12.Final.jar hibernate-entitymanager-4.1.12.Final.jar hibernate-jpa-2.0-api-1.0.1.Final.jar ...

    最新版本的Struts2+Spring4+Hibernate4框架整合

    项目原型:Struts2.3.16 + Spring4.1.1 + Hibernate4.3.6 二、 项目目的: 整合使用最新版本的三大框架(即Struts2、Spring4和Hibernate4),搭建项目架构原型。 项目架构原型:Struts2.3.16 + Spring4.1.1 + ...

    hibernate学习资料大全

    【hibernate学习资料大全】 Hibernate 是一个开源的对象关系映射(ORM)框架,它极大地简化了Java应用程序对数据库的操作。这个压缩包包含了丰富的Hibernate学习资源,旨在帮助开发者深入理解和熟练掌握这一强大的...

    Hibernate-extensions 完整安装包

    《Hibernate-Extensions全面指南》 Hibernate,作为Java领域中的一款著名对象关系映射(ORM)框架,极大地简化了数据库操作。然而,为了满足更复杂的业务需求,Hibernate还提供了丰富的扩展功能,这就是我们今天要...

    hibernate-release-4.1.4

    【描述】中的"hibernate的jar包"指的是Hibernate框架的运行库文件,这些JAR文件包含了Hibernate的所有核心API、实现和依赖库,如Hibernate Commons Annotations、Hibernate EntityManager、Hibernate Core等。...

    hibernate基础jar包

    Hibernate是一个开源的对象关系映射(ORM)框架,它允许Java开发者使用面向对象的方式来操作数据库,极大地简化了数据访问层的编程工作。这个压缩包包含了Hibernate的基础jar包,这些jar文件是开发Hibernate应用所...

    HibernateTools-3.2.4

    HibernateTools是Java开发人员在使用Hibernate ORM框架时的有力辅助工具集,主要目的是为了提高开发效率,简化数据库操作。在HibernateTools 3.2.4版本中,它包含了一系列的特性与插件,以支持更便捷地进行对象关系...

    Hibernate Spatial 4 教程

    Hibernate Spatial 4 教程 Hibernate Spatial 是一个基于 Hibernate 或 JPA 的空间几何对象数据库操作框架,提供了实现空间几何对象数据库操作的方法和配置说明。 一、 Hibernate Spatial 4 简介 Hibernate ...

    hibernate3全部jar包:hibernate3.jar.zip 下载

    Hibernate3 是一个强大的Java持久化框架,它允许开发者将数据库操作与业务逻辑解耦,使得应用程序的开发更为简便。这个“hibernate3全部jar包:hibernate3.jar.zip”包含了所有必要的库文件,方便用户一次性下载并...

    Hibernate3的依赖包

    Hibernate3是一个广泛使用的Java对象关系映射(ORM)框架,它允许开发者用面向对象的方式处理数据库操作,极大地简化了Java应用程序与数据库之间的交互。在这个"Hibernate3的依赖包"中,包含了运行Hibernate3应用...

    kingbaseV8 hibernate jdbc 驱动

    在Java开发环境中,与KingbaseV8数据库进行交互通常会用到Hibernate框架和JDBC驱动。 Hibernate是一个优秀的对象关系映射(ORM)框架,它简化了Java应用程序对数据库的操作,通过将Java对象与数据库表进行映射,...

    hibernate和MySQL的jar

    标题中的“hibernate和MySQL的jar”指的是Hibernate ORM框架与MySQL数据库之间的连接库。Hibernate是一种流行的Java对象关系映射(ORM)工具,它允许开发者使用面向对象的编程方式来操作数据库,而无需直接编写SQL...

    Hibernate 中文api 等学习资料

    标题"Hibernate 中文api 等学习资料"暗示了这是一组针对Hibernate ORM框架的中文学习资源,包括API文档和其他指南,旨在帮助用户更好地理解和使用Hibernate。 描述中的"hibernate orm框架api中文文档,学习资料,...

    hibernate.5.1.0.jar全部

    包含hibernate所有所需jar包还有一些其他包日志包、jpa支持包等: 列如:hibernate-core-5.1.0.Final.jar hibernate-ehcache-5.1.0.Final.jar hibernate-entitymanager-5.1.0.Final.jar hibernate-envers-5.1.0....

    hibernate2 升级为hibernate3的需要注意的事项

    ### Hibernate2 升级至 Hibernate3 的注意事项 #### 背景与目的 随着技术的发展,软件框架也在不断地更新迭代,以适应更多的需求并优化性能。Hibernate 作为一款流行的 Java 持久层框架,其从版本 2 升级到版本 3...

    Hibernate入门jar包

    Hibernate是一款强大的Java持久化框架,它简化了数据库与Java对象之间的交互,使开发者可以更加专注于业务逻辑而不是数据访问层的实现。本压缩包提供的是Hibernate入门所需的jar包,包括了Hibernate的核心库以及与之...

    hibernate4.3完整jar包

    Hibernate是一个开放源代码的对象关系映射框架,它对JDBC进行了非常轻量级的对象封装,使得Java程序员可以随心所欲的使用对象编程思维来操纵数据库。 Hibernate可以应用在任何使用JDBC的场合,既可以在Java的客户端...

    hibernate3必要jar包

    Hibernate3 是一个非常重要的Java持久化框架,它允许开发者将对象模型与数据库关系模型进行映射,从而简化了数据存取的过程。这个压缩包“hibernate3必要jar包”显然是针对Hibernate3版本的,已经去除了冗余的库文件...

Global site tag (gtag.js) - Google Analytics