使用Hibernate的一個完整例子

北京華園天一科技有限公司高級軟件工程師
2003 年 10 月
對象、關係的映射(ORM)是一種耗時的工作,在Java環境下,有幾種框架來表示持久數據,如實體Bean、OJB、JDO、Hibernate等。Hibernate是一種新的ORM映射工具,它不僅提供了從Java類到數據表的映射,也提供了數據查詢和恢復等機制。本文介紹怎麼在Web應用開發中配置Hibernate的環境,並且使用Hibernate來開發一個具體的實例。

閱讀本文前您需要以下的知識和工具:

本文的參考資料見 參考資料

介紹

面向對象的開發方法是當今的主流,但是同時我們不得不使用關係型數據庫,所以在企業級應用開發的環境中,對象、關係的映射(ORM)是一種耗時的工作。圍繞對象關係的映射和持久數據的訪問,在Java領域中發展起來了一些API和框架,下面分別簡單介紹。

JDBC可以說是訪問持久數據層最原始、最直接的方法。在企業級應用開發中,我們可能使用DAO(Data Access Object)模式來把數據訪問封裝起來,然後在其它的層中同一調用。這種方式的優點是運行效率最高,缺點是把DAO對象和SQL語言緊密耦合在一起使得在大項目中難以維護。但是不管怎麼說,使用JDBC來直接訪問持久數據層是當今企業級應用開發中使用最廣泛的。

實體Bean是J2EE平臺中用來表示和訪問持久數據的方式。雖然實體Bean是一種方便快捷的方法,但是在運行時我們需要額外購買EJB容器(當然,如今也有免費的EJB容器,如JBOSS),並且使用不同的應用服務器,需要重新書寫不同的部署描述,使得在不同應用服務器下移植企業級應用會帶來一些困難。

另外,在Java領域中,還有一些表示持久數據的框架,比如JDO和OJB,在這裏就不詳細介紹了。

Hibernate是一種新的ORM映射工具,它不僅提供了從Java類到數據表之間的映射,也提供了數據查詢和恢復機制。相對於使用JDBC和SQL來手工操作數據庫,使用Hibernate,可以大大減少操作數據庫的工作量。

Hibernate可以和多種Web服務器或者應用服務器良好集成,如今已經支持幾乎所有的流行的數據庫服務器(達16種)。

下面我們來介紹怎麼結合Hibernate2.0和Apache Tomcat5.0在Web應用中使用Hibernate。

配置

1、 下載安裝Tomcat,並且下載Hibernate的運行環境(主要包含一些JAR包)。

2、 把要使用的數據庫的JDBC驅動程序拷貝到%TOMCAT_HOME%/common/lib目錄下。筆者使用的是MYSQL,對應的驅動程序的JAR包爲mm.mysql-2.0.4-bin.jar。

3、 在Tomcat的Webapps目錄下新建一個Web應用,名字爲hibernate。

4、 把Hibernate提供的hibernate2.jar和一些第三方的運行庫拷貝到hibernate/WEB/INF/lib目錄下。(這些第三方的運行庫包含在下載的Hibernate lib目錄下)

5、 在%TOMCAT_HOME%/conf/server.xml中Web應用和數據源。在server.xml中加入以下的配置描述。

例程1 配置web應用




<Context path="/hibernate" docBase="hibernate" debug="0"   reloadable="true" crossContext="true"> 

<Resource name="jdbc/hibernate" auth="Container" type="javax.sql.DataSource"/>   

     <ResourceParams name="jdbc/hibernate">	

     	<parameter>		

     	    <name>factory</name>		

     		<value>org.apache.commons.dbcp.BasicDataSourceFactory</value>		

     	</parameter>

     	<parameter>		

    	    <name>driverClassName</name>		

     		     <value>org.gjt.mm.mysql.Driver</value>		

     	 </parameter>	

     	  <parameter>	

     		  	    <name>url</name>		

     		  	    <value>jdbc:mysql:///test</value>   

         </parameter>		

     	 <parameter>	

     	 	    <name>username</name>	

     	    	<value>root</value>	

     	 </parameter>		

     	<parameter>	

	        <name>password</name>		

	        <value></value>	

          </parameter>		

     	<parameter>	

		    <name>maxActive</name>	

		    <value>20</value>	

		  </parameter>

         <parameter>		

     		 <name>maxIdle</name>		

	          <value>10</value>

		  </parameter>	

		  <parameter>	

     	    <name>maxWait</name>	

		    <value>-1</value>	

		  </parameter>		

    </ResourceParams>   

  </Context>

在這裏,配置了一個名爲hibernate的Web應用,並且配置了一個數據源,數據源的JNDI名稱爲jdbc/hibernate。您需要根據情況修改數據源的鏈接屬性。

6、 下一步就是書寫Hibernate的配置描述符。可以使用XML的配置描述,也可以使用基於屬性的配置描述。在這裏使用基於XML的配置描述。在hibernate/WEB-INF/classes目錄下新建一個hibernate.cfg.xml文件。然後加入例程2所示的內容。

 


<!DOCTYPE hibernate-configuration

    PUBLIC "-//Hibernate/Hibernate Configuration DTD//EN"

    "http://hibernate.sourceforge.net/hibernate-configuration-2.0.dtd">



<hibernate-configuration>

    <session-factory>

        <property name="connection.datasource">java:comp/env/jdbc/hibernate</property>

        <property name="show_sql">false</property>

        <property name="dialect">net.sf.hibernate.dialect.MySQLDialect</property>



        <!-- Mapping files -->

        

    </session-factory>



</hibernate-configuration>

注意connection.datasource屬性必須和server.xml中配置的數據源的屬性一樣。如果不是使用MYSQL,那麼需要更改dialect屬性。

到現在,配置基本完成,下面我們來開發一個最簡單的應用。

開發持久對象、編寫映射描述

我們使用hibernate來封裝一個簡單的數據表。這個表的名字爲Courses,它有兩個字段,一個是ID,它是Courses表的主鍵;另一個是name,表示Courses的名字。在數據庫中使用以下的腳本來創建這個表:

create table Courses(CourseId varchar(32) not null, name varchar(32), constraint pk_Courses primary key (CourseId) );

接下來的任務就是爲Courses表書寫持久對象,如例程3所示。

例程3 Courses的持久對象(Courses.java)

 
package com.hellking.study.hibernate;

import java.util.Set;

/**

 *在hibernate中代表了Course表的類。

*/

public class Course 

{

   /**每個屬性和表的一個字段對應**/

   private String id;

   private String name;

   
   /**students表示course中的學生,在後面纔會用到,暫時不管**/

   private Set students;

   	
    /**屬性的訪問方法**/

	public void setId(String string) {

		id = string;

	}

	

	public String getId() {

		return id;

	}

	

	public void setName(String name)

	{

		this.name=name;

	}

	public String getName()

	{

		return this.name;

	}

	public void setStudents(Set stud)

	{

		this.students=stud;

	}

	public Set getStudents()

	{

		return this.students;

	}

}

可以看出,在Course類中也包含了兩個屬性,id和name,它的屬性和表Courses的字段是一一對應的,並且類型一致。

書寫好了持久對象,接下來的任務就是書寫對象、關係映射描述。在hibernate/WEB-INF/classes目錄下新建一個Course.hbm.xml描述文件,內容如例程4所示。

例程4 Course.hbm.xml

 

<?xml version="1.0"?>

<!DOCTYPE hibernate-mapping PUBLIC

    "-//Hibernate/Hibernate Mapping DTD 2.0//EN" 

    "http://hibernate.sourceforge.net/hibernate-mapping-2.0.dtd">

<hibernate-mapping>

    <class

        name="com.hellking.study.hibernate.Course"

        table="Courses"

        dynamic-update="false"

    >

        <id

            name="id"

            column="CourseId"

            type="string"

            unsaved-value="any"

        >

            <generator class="assigned"/>

        </id>



        <property

            name="name"

            type="string"

            update="true"

            insert="true"

            column="Name"

        />      

    </class>

</hibernate-mapping>

在Course.hbm.xml映射文件中,指定了要映射的類和映射的表,並且指定了表的各個字段和Java對象中各個字段的映射關係,比如Course對象中的id屬性對應了Courses表的courseId字段。

接下來的任務就是在hibernate.cfg.xml中指定這個映射關係。如下所示:

 

<session-factory>

…

<!-- Mapping files -->  

 <mapping resource="Course.hbm.xml"/>

</session-factory>

編寫業務邏輯

到此,我們已經封裝了一個名爲Courses的表,並且配置完成。接下來的任務就是在Web應用開發中使用它們,爲了演示在Hibernate中對數據庫的不同類型的操作,我們開發的Web應用有以下的功能:

  • 增加一個Course;

  • 刪除一個Course;

  • 按照Course的名字進行模糊搜索;

  • 查看系統中所有的Course。

雖然我們可以直接在JSP中使用hibernate,但是往往我們不這樣,而是把這些業務邏輯封裝在JavaBean中,然後在JSP中通過調用JavaBean以訪問Hibernate封裝的對象。

由於訪問通過使用hibernate有一些共性的操作,在這裏我們把這些共性的操作封裝在一個專門的類中,這樣其它的類可以繼承它,如例程5所示。

例程5 HibernateBase.java

 

package com.hellking.study.hibernate;


import net.sf.hibernate.*;

import net.sf.hibernate.cfg.*;

import java.util.*;

import java.io.IOException;

import java.io.PrintWriter;


public abstract class HibernateBase 

{

	protected SessionFactory sessionFactory;//會話工廠,用於創建會話

    protected Session session;//hibernate會話

    protected Transaction transaction; //hiberante事務

    

    public HibernateBase()throws HibernateException

    {

    	this.initHibernate();

    }

    // 幫助方法

    protected void initHibernate()

        throws HibernateException {



        // 裝載配置,構造SessionFactory對象

        sessionFactory = new Configuration().configure().buildSessionFactory();

    }

    

    /**

     *開始一個hibernate事務

     */

    protected void beginTransaction()

        throws HibernateException {



        session = sessionFactory.openSession();

        transaction = session.beginTransaction();

    }

    

    /**

     *結束一個hibernate事務。

     */

    protected void endTransaction(boolean commit)

        throws HibernateException {



        if (commit) {

            transaction.commit();

        } else {

           //如果是隻讀的操作,不需要commit這個事務。

            transaction.rollback();

        }

         session.close();

    }

}


下面編寫業務邏輯類,新建一個名爲CourseBean的JavaBean,並且CourseBean繼承HibernateBase類,代碼如例程6所示。

例程6 CourseBean.java

 

package com.hellking.study.hibernate;



import net.sf.hibernate.*;

import net.sf.hibernate.cfg.*;

import java.util.*;



/**

 *和course相關的業務邏輯

 */

public class CourseBean extends HibernateBase

{

	public CourseBean()throws HibernateException

	{

		super();

	}

	/**

	 *增加一個Course

	 */

	public void addCourse(Course st)throws HibernateException

	{

		beginTransaction();

         session.save(st);         

         endTransaction(true);

    }

    

    /**

     *查詢系統中所有的Course,返回的是包含有Course持久對象的Iterator。

     */

    public Iterator getAllCourses()throws HibernateException

    {

     	String queryString = "select courses from Course as courses";

        beginTransaction();

        Query query = session.createQuery(queryString);

        Iterator it= query.iterate();

        return it;

    }

    

    /**

     *刪除給定ID的course

     */

    public void deleteCourse(String id)throws HibernateException

    {

     	beginTransaction();    	

     	Course course=(Course)session.load(Course.class,id);    	

     	session.delete(course);

     	endTransaction(true);

     }

    

    /**

     *按course的名字進行模糊查找,返回的是包含有Course持久對象的Iterator。

     */

    public Iterator getSomeCourse(String name)throws HibernateException

    {

      	String queryString = "select c from Course as c where c.name like :name" ;

         beginTransaction();

         Query query = session.createQuery(queryString);

         query.setString("name", "%"+name+"%");

        Iterator it= query.iterate();

        return it;

    }    	

}

在CourseBean封裝了4個業務方法,你可以根據情況增加其它的業務方法。在CourseBean中,通過Hibernate來操作潛在的數據庫資源。

要保存Course數據到數據庫,可以通過:

 

session.save(Course);

方法來保存,它相當於使用在JDBC中執行以下語句:

 

Connection con=…

Statement stmt=con.createStatement();

stmt.executeUpdate("insert into courses values('"+course.getId(),+"','"+course.getName()+"')");

con.close();

可以看出,通過使用Hibernate,可以大大減少數據訪問的複雜度。

在JSP中調用業務邏輯

添加數據

CourseBean這個業務對象封裝了和Hibernate的交互關係,從而使JSP和Hibernate關係的解藕。我們來看測試主頁面的部分代碼,如例程7所示。

例程7 測試Hibernate開發的應用(course.jsp)

 

<%@ page import="java.sql.*,java.util.*" errorPage="error.jsp"%> 

<jsp:useBean id="course" class="com.hellking.study.hibernate.Course" scope="page">

<jsp:setProperty name="course" property="*"/>

</jsp:useBean>

<jsp:useBean id="courseBusiness" class="com.hellking.study.hibernate.CourseBean" scope="page"/>

<html><body><center>

<%  

  try

  {

  if(course.getId().equals(null)||course.getId().equals(""));

  else courseBusiness.addCourse(course);

  

  %>

成功添加了Course:<br>

name:<%=course.getName()%>

Id:<%=course.getId()%>

<%

}

  catch(Exception e)

  {

  }  

%> 



<hr>

<br>::增加一個course::<br>

<form action="course.jsp" method="get" name="add">

id:<input type=text name="id"><br>

name:<input type=text name="name"><br>

<input type=submit value="submit"><br>

</form>

<hr>

::按名字模糊查找::<br>

<form action="queryCourse.jsp" method="get" name="queryByName">

name:<input type=text name="name"><br>

<input type=submit value="query"><br>

</form>

<hr>

::刪除一個Course::<br>

<form action="deleteCourse.jsp" method="get" name="queryByName">

id:<input type=text name="id"><br>

<input type=submit value="delete"><br>

</form>

<hr>

<a href=viewAll.jsp>::查看所有Course::<a>

</body>

</html>

首先通過一個值對象Course(這個類正好是Hibernate使用的持久對象,這裏作爲值對象來傳遞數據)接收穫得的參數,然後CourseBean的addCourse(Course)方法把數據保存到數據庫。可以看出,通過使用Hibernate,把數據從表單中添加到數據庫非常簡單。

查詢

下面來看模糊查找的JSP代碼,如例程8所示

例程8 按名字模糊查找Course

 

<%@ page import="java.sql.*,java.util.*,com.hellking.study.hibernate.Course" errorPage="error.jsp"%> 

<jsp:useBean id="courseBusiness" class="com.hellking.study.hibernate.CourseBean" scope="page"/>

…

<% try

{

   Iterator it=courseBusiness.getSomeCourse((String)request.getParameter("name"));

   while(it.hasNext())

   {

     Course temp=(Course)it.next();

     out.println("<tr><td>"+temp.getId()+"</td>");

     out.println("<td>"+temp.getName()+"</td></tr>");

   }

  }

  catch(Exception e)

  {

    out.println(e.getMessage());

   }

%>

….

它實際上調用的是CourseBean的Iterator getSomeCourse(String name)方法。我們來回顧一下這個方法中的代碼:

 


/**

     *按course的名字進行模糊查找

     */

    public Iterator getSomeCourse(String name)throws HibernateException

    {

      	String queryString = "select c from Course as c where c.name like :name" ;

        beginTransaction();

        Query query = session.createQuery(queryString);

         query.setString("name", "%"+name+"%");

        Iterator it= query.iterate();

        return it;

    }

在查詢前,首先調用beginTransaction方法啓動新的Hibernate事務,然後創建一個Query對象,在創建這個對象時,同時指定查詢的語句。

注意,在查詢語句:

 

select c from Course as c where c.name like :name"

中,它雖然和普通的SQL語句相似,但是不同,在數據庫中,使用的表的名字是Courses,而在這個查詢語句中使用的是Course,它和持久對象的名字一致,也就是說,這個查詢的概念是查詢持久對象,而不是數據庫的記錄。

創建了查詢對象Query後,需要設置查詢的參數,它和在JDBC中PreparedStatement對象中設置參數的方法相似。通過"Iterator it= query.iterate()"語句來執行查詢,並且返回一個Iterator對象。在這裏使用了Hibernate提供的查詢機制,一般的JDBC查詢返回的是ResultSet對象,而這裏返回的是包含了CourseBean對象的Iterator。

要查詢系統中所有的Course,也同樣非常簡單,可以通過例程9所示的代碼實現。

例程9 查詢數據庫中所有的Course

 


…

<jsp:useBean id="courseBusiness" class="com.hellking.study.hibernate.CourseBean" scope="page"/>

…

<% try

{

   Iterator it=courseBusiness.getAllCourses();

   while(it.hasNext())

   {

     Course temp=(Course)it.next();

     out.println("<tr><td>"+temp.getId()+"</td>");

     out.println("<td>"+temp.getName()+"</td></tr>");

   }

  }

  catch(Exception e)

  {

    out.println(e.getMessage());

   }

%>

…

實際上調用的是CourseBean的getAllCourses方法,它和getSomeCourse方法機制一樣,就不再介紹了。

刪除數據

在JSP中,使用以下的代碼來執行刪除操作。

例程10 刪除數據庫中Courses表的記錄

 


<jsp:useBean id="courseBusiness" class="com.hellking.study.hibernate.CourseBean" scope="page"/>

…

刪除id爲:<%=request.getParameter("id")%>的course::::<br>



<% try

{

	courseBusiness.deleteCourse(request.getParameter("id"));

	out.println("刪除成功");

} 

  catch(Exception e)

  {

    out.println("不存在這個記錄");

   }

%>


我們來看CourseBean中執行刪除操作的具體代碼:
 


/**

     *刪除給定ID的course

     */

    public void deleteCourse(String id)throws HibernateException

    {

    	beginTransaction();    	

    	Course course=(Course)session.load(Course.class,id);    	

    	session.delete(course);

    	endTransaction(true);

     }

在這個方法中,首先開始一個事務,然後通過session.load(Course.class,id)方法來裝載指定ID的持久對象,接下來通過"session.delete(course)"來刪除已經裝載的course,並且結束Hibernate事務。

總結

下面總結一下使用Hibernate的開發過程:

1、 配置Hibernate(一次即可);

2、 確定數據表;

3、 創建持久對象;

4、 編寫對象和數據表的映射描述;

5、 編寫和業務邏輯。

實際上,上面的過程和使用EJB沒有什麼區別:在使用EJB時,首先當然也是配置環境,初始化數據表;然後創建實體Bean(對象於Hibernate的持久對象);接下來編寫部署描述符(ejb-jar.xml,廠商專有的部署描述),在這些部署描述符裏,指定了EJB和數據表的映射關係,如果多個實體Bean存在關聯關係,需要描述它們之間的關係,這些描述對應於Hibernate中持久對象的描述,如Course.hbm.xml;往往我們並不在應用程序中直接操作實體Bean,而是通過業務對象(如會話Bean)來操作,這裏的會話Bean可以簡單的和Hibernate中執行業務邏輯的JavaBean對應。這裏只是簡單的類比,不是絕對的,比如我們同樣可以在會話Bean中訪問Hibernate持久對象,也就是說使用Hibernate,同樣可以把業務邏輯放在會話Bean中。

通過本文的學習,相信讀者對Hibernate已經有了初步的認識,並且能夠使用Hibernate開發簡單的應用。在下一篇中,我們將學習怎麼使用Hibernate來爲複雜的數據表進行映射,並且維護它們之間的關係。

參考資料

  • http://www.apache.org 下載Tomcat。
  • Hibernate的官方網站,http://hibernate.bluemars.net/ ,包含了Hibernate最新資料。
  • Hibernate中文論壇,hibernate.fankai.com包含了Hibernate較多的參考資料。
  • 包含了Hibernate技術討論網站,www.jdon.com
  • 於Hibernate、JDO、CMP等技術的熱烈討論1:

    http://www.jdon.com/jive/thread.jsp?forum=16&thread=6062&start=0&msRange=15
  • 於Hibernate、JDO、CMP等技術的熱烈討論2:


    http://www.theserverside.com/discussion/thread.jsp?thread_id=19732
  • Hibernate2 Reference Documentation,可以從Hibernate官方網站獲得,非常好的參考資料。
  • Hibernate In Action,一本非常專業的Hibernate參考書,由Hibernate項目主要開發人員Gavin King 等著,Manning出版社出版。您可以從 http://www.theserverside.com 獲得本書的部分章節。
 

關於作者


陳亞強:北京華園天一科技有限公司高級軟件工程師,擅長J2EE技術,曾參與多個J2EE項目的設計和開發,對Web服務有很大的興趣並且有一定的項目經驗。熱愛學習,喜歡新技術,曾參與多本圖書的寫作。好交朋友,您可以通過 [email protected] 和他聯繫。

發佈了35 篇原創文章 · 獲贊 1 · 訪問量 6萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章