細心!用心!耐心!

          吾非文人,乃市井一俗人也,讀百卷書,跨江河千里,故申城一游; 一兩滴辛酸,三四年學業,五六點粗墨,七八筆買賣,九十道人情。

          BlogJava 聯系 聚合 管理
            1 Posts :: 196 Stories :: 10 Comments :: 0 Trackbacks
           Kodo EJB是一個支持對象/關系映射的框架,根據EJB3規范的要求,Kodo EJB除了支持在普通Java應用中提供輕量級的持久層框架之外,也支持在JAVA EE容器中使用滿足重量級企業應用的需求,充分利用JAVA EE容器中提供的優越特性如容器管理事務、遠程(Remote)訪問。

            基于Kodo EJB開發的應用支持使用EJB或者JCA標準接入到JAVA EE環境中:

          1. JCA
            Kodo EJB支持JCA1.0標準,因此基于Kodo EJB開發的應用可以和其他JCA資源一樣輕松的發布到JAVA EE應用服務器上。
          2. JNDI
            另外一種方式是將kodo.persistence.EntityManagerFactoryImpl的一個實例綁定到JNDI,然后通過查找JNDI的方式使用Kodo EJB應用。雖然這種方式需要開發者根據不同的JAVA EE容器編寫代碼才能完成,但是這種方式能夠保持最大限度的JAVA EE容器可移植性,而且為在那些不支持JCA標準的JAVA EE容器中使用Kodo EJB創造可能。

            本文中我們將以通過一個簡單的例子,簡單的講解和演示如何在Weblogic9上通過JNDI方式來訪問Kodo EJB應用。

          環境準備

            由于Kodo是基于注釋機制的框架,我們必須使用JDK5.0完成開發工作。所以下載、安裝Kodo之前,請確保您已經下載和安裝了JDK5.0。

            為了演示的需要,我們選擇MySQL數據庫作為持久化的目標數據庫,請大家自己到www.mysql.com下載最新的MySQL數據庫后安裝。

          安裝Kodo

            Kodo的最新版本是Kodo 4.0.0 Early Access 4,目前大家可以到http://www.solarmetric.com/去下載試用版本,下載時需要注冊,你會得到30天的License。
          將下載的壓縮文件解壓到c:/kodo4目錄下(后面將使用%KODO_HOME%來引用這個目錄),打開%KODO_HOME%/bin/kodocmd.cmd文件,將其中的KODODIR設置為您的Kodo的安裝目錄,將JDKHOME設置為Java的安裝目錄。

          安裝Weblogic9

            Kodo EJB在JAVA EE環境下運行時需要EJB容器支持EJB3標準,BEA最新發布的Weblogic9服務器就支持EJB3標準,因此我們選擇Weblogic9作為演示時使用的目標服務器。大家可以到http://www.bea.com下載Weblogic9,然后將Weblogic9安裝到自己的機器上。

            [注]本文中的實例是實現可遠程訪問的EJB實例,因此你可以將Weblogic9安裝到其他機器上,不過客戶端調用的時候需要增加一些環境變量,請參考后面的說明。

          開發Kodo EJB應用

            由于篇幅的關系,我們直接使用《Kodo EJB:符合EJB3規范的持久層框架》一文中已經創建好的例子,包括環境準備、持久化類創建、數據庫創建等工作,下面的章節中將盡量不再重復《Kodo EJB:符合EJB3規范的持久層框架》一文中已經提到的步驟,而是重點描述需要額外完成的工作。

            在EJB開發中,我們通常都采用Session Façade設計模式,因此下面的例子也使用了這種設計模式來封裝Kodo EJB應用。

            我們使用一個無狀態的Session Bean來封裝對Book類的所有操作,由于Kodo EJB應用中需要開發者完成kodo.persistence.EntityManagerFactoryImpl的實例到JNDI的綁定,我們把這部分工作訪在Session Bean中完成,下面是Session Bean的接口和實現的全部代碼,請注意代碼中增加的注釋,他們有助于您了解Kodo EJB如何工作的。

          BookBean類

            下面是BookBean類的源代碼,請大家特別注意BookBean類中的setSessionContext方法,其中的代碼將
          kodo.persistence.EntityManagerFactoryImpl的實例綁定到JNDI上。

          package org.vivianj.kodo.examples.ejb.stateless;
          import java.rmi.RemoteException;
          import java.util.Collection;
          import javax.ejb.EJBException;
          import javax.ejb.SessionBean;
          import javax.ejb.SessionContext;
          import javax.naming.InitialContext;
          import javax.naming.NamingException;
          import javax.persistence.EntityManager;
          import javax.persistence.EntityManagerFactory;
          import javax.persistence.PersistenceContextType;
          import javax.persistence.Query;
          import kodo.persistence.KodoPersistence;
          import org.vivianj.kodo.examples.beans.Book;
          /**
          * BookBean 提供Session Bean的實現類
          */
          public class BookBean implements SessionBean {
          protected SessionContext ctx;
          private EntityManagerFactory emf;
          /**
          * getBookById 根據Book對象的id屬性查找、返回符合條件的Book對象
          *
          * @param id
          *      Book對象的編號
          * @return 編號為id的Book對象
          * @throws RemoteException
          */
          public Book getBookById(int id) throws RemoteException {
          /* 獲得EJB的實體管理器 */
          EntityManager em = emf
          .createEntityManager(PersistenceContextType.EXTENDED);
          try {
          /* 開始事務 */
          em.getTransaction().begin();
          /* 處理業務 */
          Book book = em.find(Book.class, id);
          /* 結束事務 */
          em.getTransaction().commit();
          return book;
          } finally {
          /* 關閉EJB實體管理器 */
          em.close();
          }
          }
          /**
          * updateBook 更新Book對象的信息
          *
          * @param book
          *      需要更新的Book對象
          * @throws RemoteException
          */
          public void updateBook(Book book) throws RemoteException {
          /* 獲得EJB的實體管理器 */
          EntityManager em = emf.getEntityManager();
          try {
          /* 開始事務 */
          em.getTransaction().begin();
          /* 處理業務 */
          em.merge(book);
          /* 結束事務 */
          em.getTransaction().commit();
          } finally {
          /* 關閉EJB實體管理器 */
          em.close();
          }
          }
          /**
          * createBook 方法用于持久化新的Book對象
          *
          * @param book
          *      被持久化的Book對象
          * @throws RemoteException
          */
          public void createBook(Book book) throws RemoteException {
          /* 獲得EJB的實體管理器 */
          EntityManager em = emf.getEntityManager();
          try {
          /* 開始事務 */
          em.getTransaction().begin();
          /* 處理業務 */
          em.persist(book);
          /* 結束事務 */
          em.getTransaction().commit();
          } finally {
          /* 關閉EJB實體管理器 */
          em.close();
          }
          }
          public void deleteBook(Book book) throws RemoteException {
          /* 獲得EJB的實體管理器 */
          EntityManager em = emf.getEntityManager();
          try {
          /* 開始事務 */
          em.getTransaction().begin();
          /* 處理業務 */
          Query q = em.createQuery("delete from Book c where c.id = :id");
          q.setParameter("id", book.id);
          q.executeUpdate();
          /* 結束事務 */
          em.getTransaction().commit();
          } finally {
          /* 關閉EJB實體管理器 */
          em.close();
          }
          }
          public Collection getBooks(String queryString) throws RemoteException {
          /* 獲得EJB的實體管理器 */
          EntityManager em = emf.getEntityManager();
          try {
          /* 開始事務 */
          em.getTransaction().begin();
          /* 處理業務 */
          List allBooks = em.createQuery(queryString).getResultList();
          /* 結束事務 */
          em.getTransaction().commit();
          return allBooks;
          } finally {
          /* 關閉EJB實體管理器 */
          em.close();
          }
          }
          public void setSessionContext(SessionContext ctx) {
          this.ctx = ctx;
          try {
          /* 獲取JAVA EE容器上下文環境 */
          Context ejbContext = new InitialContext ();
          Object o = null;
          try{
          /* 根據JNDI查找kodo.persistence.EntityManagerFactoryImpl的實例 */
          o = ejbContext.lookup("ejb/kodo/emf");
          }catch(Exception e){
          /* 如果沒有找到,嘗試將kodo.persistence.EntityManagerFactoryIm
          pl的實例綁定到JNDI上 */
          EntityManagerFactory emfForEjb =
          Persistence.createEntityManagerFactory(null);
          ejbContext.bind("ejb/kodo/emf",emfForEjb);
          }
          if (o == null)
          {
          /* 如果沒有找到,嘗試再次查找 */
          emf = (EntityManagerFactory)ejbContext.lookup("ejb/kodo/emf");
          }
          else {
          emf = (EntityManagerFactory)o;
          }
          } catch (NamingException e) {
          throw new RuntimeException(e);
          }
          }
          public void ejbCreate() throws RemoteException {
          }
          public void ejbActivate() throws EJBException, RemoteException {
          }
          public void ejbPassivate() throws EJBException, RemoteException {
          }
          public void ejbRemove() throws EJBException, RemoteException {
          }
          public void unsetSessionContext() {
          ctx = null;
          }
          }
          

          BookHome接口

          package org.vivianj.kodo.examples.ejb.stateless;
          import java.rmi.RemoteException;
          import javax.ejb.CreateException;
          import javax.ejb.EJBHome;
          /**
          * BookHome 類提供Session Bean的home接口
          */
          public interface BookHome extends EJBHome {
          public BookRemote create() throws CreateException, RemoteException;
          }
          

          BookRemote接口

          package org.vivianj.kodo.examples.ejb.stateless;
          import java.rmi.RemoteException;
          import java.util.Collection;
          import javax.ejb.EJBObject;
          import org.vivianj.kodo.examples.beans.Book;
          /**
          * BookRemote 類提供Session Bean的remote接口
          */
          public interface BookRemote extends EJBObject {
          public Book getBookById(int id) throws RemoteException;
          public void updateBook(Book book) throws RemoteException;
          public void createBook(Book book) throws RemoteException;
          public void deleteBook(Book book) throws RemoteException;
          public Collection getBooks(String queryString) throws RemoteException;
          }
          

          編寫EJB描述符文件

            為了將該Session Bean發布到Weblogic9中,我們還需要提供兩個配置文件:ejb-jar.xml和weblogic-ejb-jar.xml,作者提供的演示實例中,這兩個文件的內容如下,開發者可以根據自己的實際環境進行調整。

          ejb-jar.xml

          <?xml version="1.0" encoding="UTF-8"?>
          <ejb-jar
          xmlns="http://java.sun.com/xml/ns/j2ee"
          xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
          xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee
          http://java.sun.com/xml/ns/j2ee/ejb-jar_2_1.xsd" version="2.1"> <display-name>Sample Kodo EJB</display-name> <enterprise-beans> <session> <ejb-name>BookEJB</ejb-name> <home>org.vivianj.kodo.examples.ejb.stateless.BookHome</home> <remote>org.vivianj.kodo.examples.ejb.stateless.BookRemote</remote> <ejb-class>org.vivianj.kodo.examples.ejb.stateless.BookBean</ejb-class> <session-type>Stateless</session-type> <transaction-type>Container</transaction-type> </session> </enterprise-beans> <assembly-descriptor> <container-transaction> <method> <ejb-name>BookEJB</ejb-name> <method-name>*</method-name> </method> <trans-attribute>Required</trans-attribute> </container-transaction> </assembly-descriptor> </ejb-jar>

          weblogic-ejb-jar.xml

          <?xml version="1.0"?>
          <weblogic-ejb-jar
          xmlns="http://www.bea.com/ns/weblogic/90"
          xmlns:j2ee="http://java.sun.com/xml/ns/j2ee"
          xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
          xsi:schemaLocation="http://www.bea.com/ns/weblogic/90
          http://www.bea.com/ns/weblogic/90/weblogic-ejb-jar.xsd">
          <weblogic-enterprise-bean>
          <ejb-name>BookEJB</ejb-name>
          <jndi-name>ejb/kodo/book</jndi-name>
          </weblogic-enterprise-bean>
          </weblogic-ejb-jar>
          

          打包部署

            基于Kodo EJB開發的EJB打包和通常的EJB打包沒有什么特別,部署過程也沒有什么特別,比較簡單的方式是使用配置工具創建新的域后,將打包好的EJB jar文件直接拷貝到域目錄下的autodeploy目錄下。

            完整地打包部署過程這里不再贅述,如果大家還不是十分熟悉,請大家參考Weblogic Server的在線幫助文檔http://edocs.bea.com/wls/docs91/index.html

            在這里要給大家介紹的是創建Weblogic9域后如何配置Kodo的兩部分內容:

          1. 安裝Kodo
            Windows平臺下創建Weblogic9域后,可以在該域所在目錄下找到bin目錄下的setDomainEnv.cmd文件。打開該文件,在其中找到set PRE_CLASSPATH=這一行,這里假如地jar文件將被加入Weblogic9服務器的啟動CLASSPATH中,因此我們把%KODO_HOME%/lib目錄下所有jar文件增加到PRE_CLASSPATH中。下面是作者的設置情況(沒有全部填寫完整,請開發者根據實際情況填寫完整)。

            set PRE_CLASSPATH= F:/OpenSource/kodo-4.0.0EA4/lib/kodo.jar; F:/OpenSource/kodo-4.0.0EA4/lib/jta-spec1_0_1.jar;F:/OpenSource/kodo-4.0.0EA4/lib/jca1.0.jar;F:/OpenSource/kodo-4.0.0EA4/lib/jdo-2.0.jar;

            可選擇的,你可以將數據庫服務器的驅動jar和其他應用中需要的jar文件放在這個變量下。
          1. 提供Kodo的license

            Kodo使用時需要提供License文件,在《Kodo EJB:符合EJB3規范的持久層框架》一文中,我們知道Kodo的License信息保存在應用的META-INF目錄中的kodo.xml文件中,和應用一起使用,可是在JAVA EE環境下,我們是將kodo.persistence.EntityManagerFactoryImpl的實例綁定到JNDI服務上,這個過程是服務器完成的,將License放在EJB包中并不能讓服務器獲取License的信息,查看Kodo的幫助文檔,也沒有看到相應的實施指南,不過作者經過測試后發現下面的方法可以完成這部分工作。
            1. 準備一個license.jar文件
              license.jar中包含META-INF目錄,該目錄下包含有提供Kodo License的kodo.xml文件。license.jar文件的結構如下:

              license.jar中包含META-INF目錄

            2. 將該jar文件加入到Weblogic域的CLASSPATH中

              請參考前面一步“安裝Kodo”的步驟將license.jar文件加入Weblogic域的CLASSPATH中。

          測試

            現在,啟動服務器,編寫段簡單的測試代碼,看看EJB是否能夠正常開始工作,下面這段代碼可以用于測試從EJB中查找符合要求的Book對象并打印它的name屬性。

          /* 提供Weblogic服務器的信息 */
          Hashtable<String,String> h = new Hashtable<String,String>();
          h.put(Context.INITIAL_CONTEXT_FACTORY,
          "weblogic.jndi.WLInitialContextFactory");
          h.put(Context.PROVIDER_URL, "t3://localhost:7001");
          /* 獲取指定服務器的上下文路徑 */
          Context ctx = new InitialContext(h);
          /* 從JNDI中獲取Session Bean的home接口 */
          Object o = ctx.lookup("ejb/kodo/book");
          BookHome home = (BookHome) PortableRemoteObject.narrow(o,BookHome.class);
          /* 獲得Session Bean的remote接口 */
          BookRemote remote = home.create();
          /* 調用業務方法 */
          System.out.println(remote.getBookById(1).name);
          

            執行客戶端的時候,請將%KODO_HOME%/lib目錄下的jar文件和Weblogic服務器安裝目錄下的server/lib/weblogic.jar文件都放在CLASSPATH中。

          總結

            在EJB3的標準中,EJB3應用既可以用于Java SE環境中,同時還可以用于Java EE環境下使用。Kodo EJB中提供了兩種不同的方式支持在Java EE環境下使用:JCA和JNDI,其中JCA方式比較簡單,而JNDI方式則能夠提供更好的靈活性。

            在本文中,作者以一個簡單的例子說明了在Weblogic9中如何將Kodo EJB中kodo.persistence.EntityManagerFactoryImpl的實例綁定到JNDI上,通過JNDI訪問Kodo EJB應用中的持久化類。

          參考資源:

            EJB3規范:http://jcp.org/aboutJava/communityprocess/pfd/jsr220/index.html
          JDO2規范:http://jcp.org/aboutJava/communityprocess/pfd/jsr243/index.html
          Kodo在線文檔:http://solarmetric.com/kodo/Documentation/4.0.0EA4/docs/full/html/index.html
          Weblogic9在線文檔:http://edocs.bea.com/wls/docs91/index.html

            下載資源kodoejb-JavaEE.jar

          posted on 2007-04-17 11:43 張金鵬 閱讀(140) 評論(0)  編輯  收藏

          只有注冊用戶登錄后才能發表評論。


          網站導航:
           
          主站蜘蛛池模板: 嘉鱼县| 乐安县| 溆浦县| 宝应县| 芜湖市| 吉林市| 永登县| 桦川县| 宜宾市| 大安市| 中牟县| 特克斯县| 寻乌县| 商水县| 汶川县| 酒泉市| 墨玉县| 吴堡县| 礼泉县| 安陆市| 蚌埠市| 左权县| 温宿县| 广宁县| 平和县| 出国| 鱼台县| 垣曲县| 横峰县| 昭苏县| 建德市| 广元市| 大兴区| 兴山县| 凤翔县| 吉木乃县| 曲阳县| 呼和浩特市| 重庆市| 乌恰县| 哈密市|