整合struts和hibernate開發嚮導

    在用struts進行項目開發的時候,你是用什麼來持久化你的數據的呢?(OJB、Castor還是hibernate?)如果沒有,我建議你看看我這篇文章,Hibernate是目前開發員普遍都很推崇的ORM,而且自帶的文檔極其豐富(我爲什麼沒有選擇OJB,恐怕這是主要的原因。)
 Struts在今年的下半年初推出了它的1.1正式版,標誌其在成熟和穩定方面有了更一步的提高。本文正是基於此版本,本來想把struts1.1中的好的特性都用到做一個比較好的例子,由於時間的原因,譬如:tiles,exception hadling,validator,多模塊、國際化等等在例子中沒有涉及,相關內容請看我在csdn的專欄文章。
(一)前期準備
a. struts1.1
http://jakarta.apache.org/struts
b. hibernate 2.0
http://hibernate.bluemars.net/
c. eclipse2.1(呵呵,我比較喜歡的一個IDE,此爲可選)
http://www.eclipse.org/
d. ant1.5.1
http://ant.apache.org/
e. jdk1.4
f. tomcat4.1

(相關網站都可以得到其最新版本)
(二)用例說明
很簡單的一個例子,關係也不復雜:貓科(Animal.java)和貓(Cat.java)。 前者對後者是一對多的關係,後者對前者是多對一的關係。
第一部分,和hibernate 相關
1. Animal類

package com.iplateau.test.hibernate.persistence;
import java.util.Set;
/**
 * Class or Interface Discription
 * @author   $Author:jack$
 * @version  $ReVision:1.0 $ <br/>
 * $Id:Animal.java 2003-8-4 16:44:02 jack Exp.
 */
public class Animal {
 private String id;
 private String name;
 private Set cats;

 public Animal() {
 }

 /**
  * @return
  */
 public Set getCats() {
  return cats;
 }
 /**
  * @return
  */
 public String getId() {
  return id;
 }
 /**
  * @return
  */
 public String getName() {
  return name;
 }
 /**
  * @param set
  */
 public void setCats(Set set) {
  cats = set;
 }
 /**
  * @param string
  */
 public void setId(String string) {
  id = string;
 }
 /**
  * @param string
  */
 public void setName(String string) {
  name = string;
 }
}

Cat類

package com.iplateau.test.hibernate.persistence;
/**
 * Class or Interface Discription
 * @author   $Author:jack$
 * @version  $ReVision:1.0 $ <br/>
 * $Id:Cat.java 2003-8-3 14:33:11 jack Exp.
 */
public class Cat {

 private String id;
 private String name;
 private String sex;
 private float weight;
  //select name form Cat as cat where cat.name=
 private Animal animal;

 public Cat() {
 }
 /**
  * @return
  */
 public String getId() {
  return id;
 }
 /**
  * @return
  */
 public String getName() {
  return name;
 }
 /**
  * @return
  */
 public String getSex() {
  return sex;
 }
 /**
  * @return
  */
 public float getWeight() {
  return weight;
 }
 /**
  * @param string
  */
 public void setId(String string) {
  id = string;
 }
 /**
  * @param string
  */
 public void setName(String string) {
  name = string;
 }
 /**
  * @param c
  */
 public void setSex(String c) {
  sex = c;
 }
 /**
  * @param f
  */
 public void setWeight(float f) {
  weight = f;
 }
 /**
  * @return
  */
 public Animal getAnimal() {
  return animal;
 }
 /**
  * @param animal
  */
 public void setAnimal(Animal animal) {
  this.animal = animal;
 }
}
   這兩個類的代碼可以在本文後邊的代碼包中找到,我之所以在此列出來的原因,想讓大家知道,可持久化的類和平常你熟悉的java類對象沒有什麼不同:屬性、get、set方法。
"透明"就"透明"在這裏了吧。

   Hibernate爲每一個可持久的類配備一個xml格式的mapping文件,他們可以通過hibernate自帶的或其他的工具生成(反過來也一樣:先有mapping文件,然後生成java代碼),剛開始學習的時候最好用手寫來熟悉。

Animal.hbm.xml

<?xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD//EN"                                   "http://hibernate.sourceforge.net/hibernate-mapping-2.0.dtd">
<hibernate-mapping>

    <class name="com.iplateau.test.hibernate.persistence.Animal" table="ANIMAL">
        <!-- A 32 hex character is our surrogate key. It's automatically
            generated by Hibernate with the UUID pattern. -->
        <id name="id" type="string" unsaved-value="null" >
            <column name="ANIMAL_ID" sql-type="char(32)" not-null="true"/>
            <generator class="uuid.hex"/>
        </id>
        <!-- A cat has to have a name, but it shouldn' be too long. -->
        <property name="name">
            <column name="NAME" sql-type="varchar(16)" not-null="true"/>
        </property>       
    <set name="cats" cascade="all" inverse="true" lazy="true">
      <key column="ANIMAL_ID"/>
      <one-to-many class="com.iplateau.test.hibernate.persistence.Cat"/>
    </set>
    </class>
</hibernate-mapping>

Cat.hbm.xml

<?xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD//EN"                                   "http://hibernate.sourceforge.net/hibernate-mapping-2.0.dtd">
<hibernate-mapping>
    <class name="com.iplateau.test.hibernate.persistence.Cat" table="CAT">
        <!-- A 32 hex character is our surrogate key. It's automatically
            generated by Hibernate with the UUID pattern. -->
        <id name="id" type="string" unsaved-value="null" >
            <column name="CAT_ID" sql-type="char(32)" not-null="true"/>
            <generator class="uuid.hex"/>
        </id>
        <!-- A cat has to have a name, but it shouldn' be too long. -->
        <property name="name">
            <column name="NAME" sql-type="varchar(16)" not-null="true"/>
        </property>
        <property name="sex"/>
        <property name="weight"/>       
        <many-to-one name="animal" class="com.iplateau.test.hibernate.persistence.Animal" column="ANIMAL_ID"/>
    </class>
</hibernate-mapping>

  具體mapping文件的格式以及詞法(仔細看還是滿簡單的),請參照hibernate自帶的文檔,絕對詳細!

  好了,通常在完成mapping文件以後,就可以利用hibernate自帶的工具生成建表語句,這時要用到ant來幫助你完成。{見附代碼包中build.xml},以下爲建表片斷:
……
……
    <target name="initdb.postgres"
            description="Generates the database schema for Mssql server."
            depends="compile">
        <pathconvert refid="hibernate.mapping.files"
      property="hibernate.mappings" pathsep=" "/>
        <java classname="net.sf.hibernate.tool.hbm2ddl.SchemaExport" fork="true">
       
            <!-- Hibernate Properties -->
            <jvmarg value="-Dhibernate.dialect=${db.dialect}"/>
            <jvmarg value="-Dhibernate.connection.driver_class=${db.driver}"/>
            <jvmarg value="-Dhibernate.connection.url=${db.url}"/>
            <jvmarg value="-Dhibernate.connection.username=${db.user}"/>
            <jvmarg value="-Dhibernate.connection.password=${db.password}"/>       
       
            <classpath refid="project.class.path" />
            <arg line="${hibernate.args}"/>
            <arg line="${hibernate.mappings}"/>
        </java>
    </target>
……
……

第二部分,整合struts,運行實例
   在第一部分,我們用hibernate對我們的數據對象進行持久化,那麼hibernate如何操縱我們的數據的呢?,粗略的說就是創建系統SessionFactory(通常只創建一次),然後實例一個Session,通常我們直接對數據進行操作都是通過Session來完成,它就象jdo中的persistenceManager,而SessionFactory呢,就象jdo的persistenceManagerFactory。兩者之間的關係不管是jdo還是hibernate都是相同的。

  下面說一下SessionFactory的創建方法,大概有三種方法(詳細請看hibernate的手冊):通過*.class;通過*.hbm.xml;通過hibernate.cfg.xml;我比較傾向於用最後一種方法,本例就是採用hibernate.cfg.xml來創建SessionFactory以及數據庫連接的。

Hibernate.cfg.xml

<?xml version='1.0' encoding='utf-8'?>
<!DOCTYPE hibernate-configuration PUBLIC "-//Hibernate/Hibernate Configuration DTD//EN"
                                         "http://hibernate.sourceforge.net/hibernate-configuration-2.0.dtd">

<hibernate-configuration>

    <session-factory>

        <property name="hibernate.connection.driver_class">com.microsoft.jdbc.sqlserver.SQLServerDriver</property>
        <property name="hibernate.connection.url">jdbc:microsoft:sqlserver://yourserver;DatabaseName=yourdatabase;SelectMethod=Cursor</property>
        <property name="hibernate.connection.username">your account</property>
        <property name="hibernate.connection.password">your password</property>
        <property name="hibernate.connection.pool.size">20</property>
        <property name="hibernate.dialect">net.sf.hibernate.dialect.SybaseDialect</property>
        <!-- Mapping files -->
        <mapping resource="com/iplateau/test/hibernate/persistence/Cat.hbm.xml"/>
  <mapping resource="com/iplateau/test/hibernate/persistence/Animal.hbm.xml"/>
    </session-factory>

</hibernate-configuration>

   OK,確定了創建方法以後我們就可以應用了,應爲SessionFactory通常只創建一次,所以推薦在系統初始化的建立,那麼可以利用struts的plugin機制來完成這一任務,見:

Com.iplateau.test.hibernate.InitHibernateDataStore.java
(hibernate手冊也推薦在struts中用這種方法創建)

  關於struts的擴展機制請參見我的另一篇文章《擴展你的struts》


package com.iplateau.test.hibernate;

import java.io.File;
import java.util.List;
import java.net.URL;
import javax.naming.Context;
import javax.naming.InitialContext;
import javax.naming.NamingException;

import net.sf.hibernate.SessionFactory;
import net.sf.hibernate.cfg.Configuration;

import javax.servlet.ServletException;

import org.apache.struts.action.ActionServlet;
import org.apache.struts.action.PlugIn;
import org.apache.struts.config.ModuleConfig;

/**
 * Class or Interface Discription
 * @author   $Author:jack$
 * @version  $ReVision:1.0 $ <br/>
 * $Id:InitHibernateDataStore.java 2003-8-3 14:08:50 jack Exp.
 */
public class InitHibernateDataStore implements PlugIn {

 private String _configFilePath = "/hibernate.cfg.xml";
 private SessionFactory _factory = null;
 static String contextFactory =
  "com.sun.jndi.rmi.registry.RegistryContextFactory";
  
 private Context ctx = null;
 /* (non-Javadoc)
  * @see org.apache.struts.action.PlugIn#destroy()
  */
 public void destroy() {
  // TODO Auto-generated method stub
  try {
   if (ctx != null) {
    ctx.close();
    ctx = null;
   }
  } catch (NamingException ex) {
   ex.printStackTrace();
  }
 }

 /* (non-Javadoc)
  * @see org.apache.struts.action.PlugIn#init(org.apache.struts.action.ActionServlet, org.apache.struts.config.ModuleConfig)
  */
 public void init(ActionServlet arg0, ModuleConfig arg1)
  throws ServletException {
  // TODO Auto-generated method stub
  Configuration configuration = null;
  URL configFileURL = null;
  System.out.println("4385738758237482173841847382374823742");
  try {
   configFileURL =
    InitHibernateDataStore.class.getResource(_configFilePath);
   configuration = (new Configuration()).configure(configFileURL);
   _factory = configuration.buildSessionFactory();

   Context ctx = new InitialContext();
   ctx.addToEnvironment(
    javax.naming.Context.INITIAL_CONTEXT_FACTORY,
    contextFactory);

   ctx.bind("hibernate_connection_factory", _factory);

  } catch (Exception e) {

  }
 }
}

在創建sessionFactory以後,採用jndi來取得:
SessionFactory sf =
    (SessionFactory) inttex.lookup("hibernate_connection_factory");
{見com.iplateau.test.hibernate.DBManager.java}

數據庫的中文問題,我採用filter來解決,效果還不錯,具體參看web.xml以及com.iplateau.test.hibernate.SetEncodingFilter.java

小節:
 總的來說,把struts和hibernate這兩種在業內比較推崇的開源技術相結合,在項目開發中不管是從效率上還是易維護上都是完美的結合。
 希望從這個簡單的例子中給你更多的提示,同時也希望你提出好的建議,通過[email protected]和我聯繫。
 附上代碼包:
 http://www.ifreeway.com.cn/download/strutsInHibernate.rar
 (兩星期內有效,過後可能會刪除)
 
參考資源:
 hibernate參考手冊(hibernate發佈包中自帶)
 hibernate example
 http://sourceforge.net/project/showfiles.php?group_id=40712
 Introduction to hibernate
 http://www.theserverside.com/resources/article.jsp?l=Hibernate

 同時http://www.chinaxp.orghttp://www.jdon.com都有很多關於hibernate的討論。


 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章