當(dāng)Spring遇到Hibernate的時(shí)候
1.介紹
最近我參與了北美一家銀行的審計(jì)系統(tǒng)的后臺(tái)軟件設(shè)計(jì)和開(kāi)發(fā)工作,除了使用Web Service以外,我們集成了Spring和Hibernate,通過(guò)Spring的HibernateTemplate實(shí)現(xiàn)了對(duì)數(shù)據(jù)庫(kù)數(shù)據(jù)的存取。眾所周知Hibernate是一種廣泛應(yīng)用的一種強(qiáng)大的數(shù)據(jù)持久層技術(shù), 另一方面Spring作為支持IOC的依賴注入框架,其優(yōu)點(diǎn)是能夠非常好地集成大多數(shù)主流技術(shù)。 本文我們將討論如何集成 Spring 和Hibernate.
2.Spring和Hibernate基礎(chǔ)
在我們實(shí)際進(jìn)入集成這兩個(gè)技術(shù)之前,讓我們理解這種集成需求,大家都知道Hibernate是一種位于應(yīng)用和數(shù)據(jù)庫(kù)之間的強(qiáng)大的ORM 工具。 它可使應(yīng)用以獨(dú)立平臺(tái)的方式從各種數(shù)據(jù)庫(kù)訪問(wèn)數(shù)據(jù),對(duì)應(yīng)用來(lái)說(shuō)沒(méi)有必要依賴于低級(jí)DBC 細(xì)節(jié)如管理連接, 處理statements 和result sets. 所有訪問(wèn)特定數(shù)據(jù)源的細(xì)節(jié)很容易在Xml文件中配置,另一個(gè)好處是Hibernate和J2SE和J2EE應(yīng)用兼容得很好。
使用Hibernate 問(wèn)題之一是使用Hibernate Framework訪問(wèn)數(shù)據(jù)庫(kù)的客戶應(yīng)用必須依賴Hibernate APIs如Configuration, SessionFactory and Session. 這些個(gè)對(duì)象在應(yīng)用代碼中持續(xù)擴(kuò)散。而且應(yīng)用代碼必須用手工維護(hù)和管理這些對(duì)象。但是在Spring的環(huán)境,業(yè)務(wù)對(duì)象通過(guò)IOC的幫助下是能夠通過(guò)配置完成的,簡(jiǎn)單地說(shuō),一個(gè)對(duì)象狀態(tài)能夠從應(yīng)用代碼中分離。意思是現(xiàn)在使用Hibernate 對(duì)象作為Spring Beans是可能的,他們能夠得到Spring提供的所有方便。
3.集成Sample
我們沒(méi)有打算研究Spring包里集成的各種API,我們想通過(guò)實(shí)例源碼來(lái)學(xué)習(xí)和理解這些APIs. 下列部分包括了集成Spring-Hiberante 的每個(gè)步驟并附有詳細(xì)的解釋。
3.1.創(chuàng)立數(shù)據(jù)庫(kù)
下面的應(yīng)用使用Oracle數(shù)據(jù)庫(kù)。 安裝后, 先用管理員身份登錄和創(chuàng)建用戶schema,username and password,再以用戶身份登錄后去SQL command 或SQL Script運(yùn)行以下SQL statement :create table Activitylog(id varchar(10), name varchar(20), taskcode varchar(3), tasktime timestamp);現(xiàn)在一個(gè)空表已創(chuàng)立。
3.2.ActivityLog類
現(xiàn)在讓我們創(chuàng)建叫做Activitylog POJO代碼用于存儲(chǔ)從Activitylog表取出的數(shù)據(jù),此類的設(shè)計(jì)使表'Activitylog'的列名將被映射到Java類Activitylog適當(dāng)類型的變量名。 可使用ant工具或plug-in產(chǎn)生,Activitylog類完整的代碼列舉如下:
- package com.xxx.audit.pojo;
- public class Activitylog {
- private String id;
- private String name;
- private String taskcode;
- private Timestamp tasktime;
- public Activitylog () {
- }
- public String getId(){
- return id;
- }
- public void setId(String id){
- this.id = id;
- }
- public String getName(){
- return name;
- }
- public void setName(String name){
- this.name = name;
- }
- public String getTaskcode(){
- return taskcode;
- }
- public void setTaskcode(String taskcode){
- this.taskcode = taskcode;
- }
- public Timestamp getTasktime(){
- return tasktime;
- }
- public void setTasktime(Timestamp tasktime){
- this.tasktime = tasktime;
- }
- public String toString(){
- return "Id = " + id + ", Name = " + name + ", Taskcode = "
- + taskcode + ", Tasktime = " + tasktime;
- }
- }
3.3.創(chuàng)建Hibernate Mapping 文件
我們已經(jīng)在數(shù)據(jù)庫(kù)中創(chuàng)建了'Activitylog' 表和一個(gè)在應(yīng)用層相應(yīng)的Java類 class,我們需要?jiǎng)?chuàng)建Hibernate Mapping文件去把'Activitylog' 表映射到'Activitylog' Java 類,'Activitylog' 表的列名映射到'Activitylog' Java 類的變量,可使用ant工具或plug-in產(chǎn)生,讓我們看看Hibernate Mapping文件:
- Activitylog.hbm.xml
- <?xml version="1.0"?>
- <!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
- "http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
- <hibernate-mapping>
- <class name="com.xxxx.audit.pojo.activitylog" table=" activitylog ">
- <id name="id" column="Id">
- <generator class="assigned"/>
- </id>
- <property name="name">
- <column name="Name"/>
- </property>
- <property name="taskcode">
- <column name="Taskcode"/>
- </property>
- <property name="tasktime">
- <column name="Tasktime"/>
- </property>
- </class>
- </hibernate-mapping>
注意映射文件是一個(gè)Xml文件并且它的名字是Activitylog.hbm.xml. 文件名中'hbm' 代表它是Hibernate映射文件。 class 標(biāo)記定義了數(shù)據(jù)庫(kù)表和Java類之間的映射。 'name' 屬性必須指向一個(gè)全路徑的Java類名,反之表屬性必須指向數(shù)據(jù)庫(kù)表,接下來(lái)的標(biāo)記定義了列名和對(duì)應(yīng)java變量之間的映射,'id' 標(biāo)記作為主鍵定義了一行的標(biāo)識(shí),property標(biāo)記有一個(gè)屬性叫'name' 指向Java變量名, 接下來(lái)的是它映射的數(shù)據(jù)庫(kù)表的列名。
3.4.創(chuàng)立Spring配置文件
這部分處理配置各種Spring需要的信息。 在Spring全部的業(yè)務(wù)對(duì)象在Xml文件中配置,配置的業(yè)務(wù)對(duì)象叫做 Spring Beans.這些Spring Beans可通過(guò)IOC維護(hù),讓我們定義一個(gè)data source如下:
- spring-hibernate.xml
- <?xml version="1.0" encoding="UTF-8"?>
- <beans xmlns="http://www.springframework.org/schema/beans"
- xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
- xsi:schemaLocation="http://www.springframework.org/schema/beans
- http://www.springframework.org/schema/beans/spring-beans-2.0.xsd">
- <bean id="myDataSource" class="org.apache.commons.dbcp.BasicDataSource" >
- <property name="driverClassName" value="oracle.jdbc.driver.OracleDriver"/>
- <property name="url" value="jdbc:oracle:thin:@localhost:1521:XE"/>
- <property name="username" value="root"/>
- <property name="password" value="root"/>
- </bean>
- ?
- </beans>
上面bean定義了一個(gè)類型'org.apache.commons.dbcp.BasicDataSource'的data-source. 更重要的是, 它定義了各種訪問(wèn)數(shù)據(jù)庫(kù)所需要的連接屬性。
4.總結(jié)
本文討論了Spring 和Hibernate的集成。 既討論了需求也討論了它帶來(lái)的好處。 通過(guò)一個(gè)詳細(xì)的實(shí)例一步步清晰地演示了集成工作。
【編輯推薦】