一、首先需要把Hibernate 用到的jar包和配置文件都放到Weblogic能夠搜索到的CLASSPATH路徑上。單單這一步就有很多人很迷茫,其實去仔細看看Weblogic的啟動腳本文件startWeblogic.cmd和startWLS.cmd,我想大部分人都知道該怎麼配置了。
我機器上的有個Hibernate的項目,在D: estoracle目錄下,該目錄下的結構是:
D: estoraclelib 放置hibernate的所有jar包
D: estoraclesrc 放置源代碼
D: estoracleclasses 編譯好的代碼和hibernate的配置文件(hibernate.properties, log4j.properties, cache.ccf)
現在需要把D: estoraclelib目錄下那些jar文件和D: estoracleclasses目錄都放置到Weblogic的 CLASSPATH裡面去,所以修改mydomain裡面的Weblogic啟動腳本startWeblogic.cmd,在啟動Weblogic之前,插入設置CLASSPATH的命令,如下:
@rem set hibernate classpath
set HIBERNATE_LIB=D: estoraclelib
set HIBERNATE_CLASSES=D: estoracleclasses
set CLASSPATH=%CLASSPATH%;%HIBERNATE_LIB%cglib-asm.jar;%HIBERNATE_LIB%commons-beanutils.jar;
%HIBERNATE_LIB%commons-collections.jar;%HIBERNATE_LIB%commons-lang.jar;
%HIBERNATE_LIB%commons-logging.jar;%HIBERNATE_LIB%dom4j-full.jar;
%HIBERNATE_LIB%hibernate2.jar;%HIBERNATE_LIB%jcs.jar;
%HIBERNATE_LIB%log4j-1.2.8.jar;%HIBERNATE_LIB%odmg.jar;
%HIBERNATE_LIB%jta.jar;%HIBERNATE_CLASSES%;
下面一行,就是本來腳本裡面的啟動命令:
@rem Call Weblogic Server
call "C:eaweblogic700serverinstartWLS.cmd"
二、在Weblogic上配置 Oracle數據庫的連接池,這一步本來和Hibernate無關,但是如果你想要使用EJB,想要使用JTA,那麼必須使用Weblogic提供的連接池,而不能使用Hibernate自帶的連接池,或者其它第三方連接池,否則容器將無法管理數據庫事務。這一步很簡單,就是在Weblogic Console裡面配置Connection Pool和TxData Source,我的TxDataSource取名稱為“mypool”
三、修改hibernate.properties。使用Weblogic的連接池,而不是自帶的連接池。我修改的是D: estoracleclasseshibernate.properties,增加如下行:
hibernate.dialect net.sf.hibernate.dialect.OracleDialect
hibernate.connection.datasource mypool
hibernate.connection.provider_class net.sf.hibernate.connection.DatasourceConnectionProvider
hibernate.session_factory_name hibernate.session_factory
注意最後一行,這是使用 Hibernate來綁定JNDI給JNDI起的名稱,本來應該是hibernate/session_factory,但是Weblogic要求改為. 號,不過在程序中lookup的時候還是要寫hibernate/session_factory
另外提到一點的是
hibernate.jdbc.fetch_size 50
hibernate.jdbc.batch_size 25
分別對數據庫查詢和插入有很大的性能影響,調節這兩個選項可以得到最好的性能。
為了保證SessionFactory實例的預創建,使用Weblogic的T3StartUpDef接口創建一個StartUp類,在Weblogic啟動的時候運行:
package com.javaeye;
import java.util.Hashtable;
import weblogic.common.T3StartupDef;
import weblogic.common.T3ServicesDef;
import net.sf.hibernate.cfg.Configuration;
import net.sf.hibernate.SessionFactory;
public class HibernateStartUp implements T3StartupDef {
public void setServices(T3ServicesDef services) {}
public String startup(String name, Hashtable args) throws Exception {
SessionFactory sf = new Configuration().configure().buildSessionFactory();
return "Hibernate Startup completed successfully";
}
}
代碼非常簡單,其實就是確保預先運行
SessionFactory sf = new Configuration().configure().buildSessionFactory();
把sf創建出來,而Hibernate會自行調用一系列類方法,把sf綁定到Weblogic的的JNDI樹下的hibernate/session_factory路徑中。
4、編譯HibernateStartUp.java
編譯這個源代碼的時候需要注意的是,要把weblogic.jar包和Hibernate所有的相關包和配置文件導入。我是把這個源代碼放到D: estoraclesrc目錄下的,用早已編寫好的ant腳本運行一下就編譯好了,並且編譯好的 class文件被放置到D: estoracleclasses目錄下,該目錄已經被加入到Weblogic的CLASSPATH裡面,因此很省事。
五、配置StartUp類
啟動Weblogic,打開Console控制台,在左邊的Applet樹上找到StartUp & Shutdown,然後在右邊點擊“Configure a new Startup Class...”,在Name框裡面隨便填寫,在ClassName裡面填寫你編寫的StartUp類,我填寫的是 com.javaeye.HibernateStartUp,然後點擊“Apply”。然後切換到Target這選項卡,在Target-Server左邊的 Avaiable框裡面選擇“myserver”,點擊右箭頭,把它挪到右邊的“Chosen”框裡面去,最後再點擊一下“Apply"按鈕。如果此時 Weblogic的DOS窗口裡面沒有出錯信息,那麼應該已經配置成功了。
六、現在關閉Weblogic,再重新運行 startWelogic.cmd,啟動Weblogic,觀察DOS窗口的輸出信息,可以看到Hibernate的初始化信息一屏屏的滾動輸出,證明已經配置成功。現在再打開Console控制台,點擊左邊Applet樹中的Servers|myserver,然後可以在右邊最下面找到“View JNDI tree ”,點擊它,會打開一個浏覽器窗口,顯示JNDI樹,這時你可以看到一個名稱為hibernate的JNDI對象,在左邊的Applet樹中點擊它,看右邊的詳細信息,我的機器上的信息如下:
Bind Name: hibernate
Object Class: net.sf.hibernate.impl.SessionFactoryImpl
Object Hash Code: 454492
Object To String: net.sf.hibernate.impl.SessionFactoryImpl@6ef5c
完全正確!
最後你可以隨意在EJB或者Servlet/JSP裡面使用JND查找來獲得SessionFactory了。
例如:
Context ctx = new InitialContext();
SessionFactory sf = (SessionFactory) ctx.lookup("hibernate/session_factory");
請注意:上述代碼只能在WebLogic容器內運行,而不能在WebLogic容器外運行。因為SessionFactory並沒有實現序列化接口,因此當客戶端程序(在另一個單獨的JVM中運行)遠程訪問WebLogic JNDI,企圖將SessionFactory序列化到本地,肯定會失敗。但即使SessionFactory實現序列化接口,由於它不是一個可以支持RMI的對象,仍然無法在WebLogic容器外正常調用。
與此不同的是,WebLogic本身的DataSource,EJB,JMS等等都是支持RMI的(前提條件是WebLogic相應的jar要有),所以你可以在WebLogic外面lookup,並且使用它。