| EJB核心技術(shù)及其應用 |
| 2005-08-29 來(lái)源:CSDN 作者:周波 |
| 文章摘要: EJB的全稱(chēng)是Enterprise java bean。是JAVA中的商業(yè)應用組件技術(shù)。EJB結構中的角色 EJB 組件結構是基于組件的分布式計算結構,是分布式應用系統中的組件... 一、EJB技術(shù)簡(jiǎn)介 EJB的全稱(chēng)是Enterprise java bean。是JAVA中的商業(yè)應用組件技術(shù)。EJB結構中的角色 EJB 組件結構是基于組件的分布式計算結構,是分布式應用系統中的組件。 一個(gè)完整的基于EJB的分布式計算結構由六個(gè)角色組成,這六個(gè)角色可以由不同的開(kāi)發(fā)商提供,每個(gè)角色所作的工作必須遵循Sun公司提供的EJB規范,以保證彼此之間的兼容性。這六個(gè)角色分別是EJB組件開(kāi)發(fā)者(Enterprise Bean Provider) 、應用組合者(Application Assembler)、部署者(Deployer)、EJB 服務(wù)器提供者(EJB Server Provider)、EJB 容器提供者(EJB Container Provider)、系統管理員(System Administrator): 二、EJB中各角色的分析 1、EJB組件開(kāi)發(fā)者(Enterprise Bean Provider) EJB組件開(kāi)發(fā)者負責開(kāi)發(fā)執行商業(yè)邏輯規則的EJB組件,開(kāi)發(fā)出的EJB組件打包成ejb-jar文件。EJB組件開(kāi)發(fā)者負責定義EJB的remote和home接口,編寫(xiě)執行商業(yè)邏輯的EJB class,提供部署EJB的部署文件(deployment descriptor)。部署文件包含EJB的名字,EJB用到的資源配置,如JDBC等。EJB組件開(kāi)發(fā)者是典型的商業(yè)應用開(kāi)發(fā)領(lǐng)域專(zhuān)家。 EJB組件開(kāi)發(fā)者不需要精通系統級的編程,因此,不需要知道一些系統級的處理細節,如事務(wù)、同步、安全、分布式計算等。 2、應用組合者(Application Assembler) 應用組合者負責利用各種EJB組合一個(gè)完整的應用系統。應用組合者有時(shí)需要提供一些相關(guān)的程序,如在一個(gè)電子商務(wù)系統里,應用組合者需要提供JSP(Java Server Page)程序。 應用組合者必須掌握所用的EJB的home和remote接口,但不需要知道這些接口的實(shí)現。 3、部署者(Deployer) 部署者負責將ejb-jar文件部署到用戶(hù)的系統環(huán)境中。系統環(huán)境包含某種EJB Server和EJB Container。部署者必須保證所有由EJB組件開(kāi)發(fā)者在部署文件中聲明的資源可用,例如,部署者必須配置好EJB所需的數據庫資源。 部署過(guò)程分兩步:部署者首先利用EJB Container提供的工具生成一些類(lèi)和接口,使EJB Container能夠利用這些類(lèi)和接口在運行狀態(tài)管理EJB。 部署者安裝EJB組件和其他在上一步生成的類(lèi)到EJB Container中。 部署者是某個(gè)EJB運行環(huán)境的專(zhuān)家。 某些情況下,部署者在部署時(shí)還需要了解EJB包含的業(yè)務(wù)方法,以便在部署完成后,寫(xiě)一些簡(jiǎn)單的程序測試。 4、EJB 服務(wù)器提供者(EJB Server Provider) EJB 服務(wù)器提供者是系統領(lǐng)域的專(zhuān)家,精通分布式交易管理,分布式對象管理及其它系統級的服務(wù)。EJB 服務(wù)器提供者一般由操作系統開(kāi)發(fā)商、中間件開(kāi)發(fā)商或數據庫開(kāi)發(fā)商提供。 在目前的EJB規范中,假定EJB 服務(wù)器提供者和EJB 容器提供者來(lái)自同一個(gè)開(kāi)發(fā)商,所以,沒(méi)有定義EJB 服務(wù)器提供者和EJB容器提供者之間的接口標準。 5、EJB 容器提供者(EJB Container Provider) EJB 容器提供者提供以下功能: 提供EJB部署工具為部署好的EJB組件提供運行環(huán)境 。EJB容器負責為EJB提供交易管理,安全管理等服務(wù)。 EJB 容器提供者必須是系統級的編程專(zhuān)家,還要具備一些應用領(lǐng)域的經(jīng)驗。EJB 容器提供者的工作主要集中在開(kāi)發(fā)一個(gè)可伸縮的,具有交易管理功能的集成在EJB 服務(wù)器中的容器。EJB 容器提供者為EJB組件開(kāi)發(fā)者提供了一組標準的、易用的API訪(fǎng)問(wèn)EJB 容器,使EJB組件開(kāi)發(fā)者不需要了解EJB服務(wù)器中的各種技術(shù)細節。 EJB容器提供者負責提供系統監測工具用來(lái)實(shí)時(shí)監測EJB容器和運行在容器中的EJB組件狀態(tài)。 6、系統管理員(System Administrator) 系統管理員負責為EJB服務(wù)器和容器提供一個(gè)企業(yè)級的計算和網(wǎng)絡(luò )環(huán)境。 系統管理員負責利用EJB 服務(wù)器和容器提供的監測管理工具監測EJB組件的運行情況。 三、EJB的體系結構: EJB分布式應用程序是基于對象組件模型的,低層的事務(wù)服務(wù)用了API技術(shù)。EJB技術(shù)簡(jiǎn)化了用JAVA語(yǔ)言編寫(xiě)的企業(yè)應用系統的開(kāi)發(fā),配置。EJB技術(shù)定義了一組可重用的組件:Enterprise Beans。你可以利用這些組件,象搭積木一樣的建立你的分布式應用程序。當你把代碼寫(xiě)好之后,這些組件就被組合到特定的文件中去。每個(gè)文件有一個(gè)或多個(gè)Enterprise Beans,在加上一些配置參數。最后,這些Enterprise Beans被配置到一個(gè)裝了EJB容器的平臺上??蛻?hù)能夠通過(guò)這些Beans的home接口,定位到某個(gè)beans,并產(chǎn)生這個(gè)beans的一個(gè)實(shí)例。這樣,客戶(hù)就能夠調用Beans的應用方法和遠程接口。 EJB服務(wù)器作為容器和低層平臺的橋梁管理著(zhù)EJB容器和函數。它向EJB容器提供了訪(fǎng)問(wèn)系統服務(wù)的能力。例如:數據庫的管理和事務(wù)的管理,或者對于其它的Enterprise的應用服務(wù)器。所有的EJB 實(shí)例都運行在EJB容器中?! ∪萜魈峁┝讼到y級的服務(wù),控制了EJB的生命周期。EJB中的有一些易于使用的管理工具如:Security--配置描述器(The Deployment descriptor)定義了客戶(hù)能夠訪(fǎng)問(wèn)的不同的應用函數。容器通過(guò)只允許授權的客戶(hù)訪(fǎng)問(wèn)這些函數來(lái)達到這個(gè)效果。Remote Connectivity--容器為遠程鏈接管理著(zhù)低層的通信issues,而且對Enterprise Beas的開(kāi)發(fā)者和客戶(hù)都隱藏了通信細節。EJB的開(kāi)發(fā)者在編寫(xiě)應用方法的時(shí)候,就象是在條用本地的平臺一樣的??蛻?hù)也不清楚他們調用的方法可能是在遠程被處理的。Life Cycle managment--客戶(hù)簡(jiǎn)單的創(chuàng )建一個(gè)Enterprise beans的實(shí)例,并通常取消一個(gè)實(shí)例。而容器管理著(zhù)Enterprise Beans的實(shí)例,使Enterprise Beans實(shí)現最大的效能和內存利用率。容器能夠這樣來(lái)激活和使Enterprise Beans失效,保持眾多客戶(hù)共享的實(shí)例池。等等?! rasction management-配置描述器定義了Enterprise beans 的事務(wù)處理的需求。容器管理著(zhù)那些管理分布式事務(wù)處理的復雜的issues。這些事務(wù)可能要在不同的平臺之間更新數據庫。容器使這些事務(wù)之間互相獨立,互不干擾。保證所有的更新數據庫都是成功發(fā)生的,否者,就回滾到事務(wù)處理之前的狀態(tài)。 EJB 組件是基于分布式事務(wù)處理的企業(yè)級應用程序的組件。所有的EJB都有如下的特點(diǎn):EJB包含了處理企業(yè)數據的應用邏輯。定義了EJB的客戶(hù)界面。這樣的界面不受容器和服務(wù)器的影響。于是,當一個(gè)EJB被集合到一個(gè)應用程序中去時(shí),不用更改代碼和重新編譯。EJB能夠被定制 各種系統級的服務(wù),例如安全和事務(wù)處理的特性,都不是屬于EJB類(lèi)的。而是由配置和組裝應用程序的工具來(lái)實(shí)現。 有兩種類(lèi)型的EJB: Session beans 和 entity beans.Session beans是一種作為單用戶(hù)執行的對象。作為對遠程的任務(wù)請求的相應,容器產(chǎn)生一個(gè)Session beans 的實(shí)例。一個(gè)Session beans有一個(gè)用戶(hù).從某種程度上來(lái)說(shuō),一個(gè)Session bean 對于服務(wù)器來(lái)說(shuō)就代表了它的那個(gè)用戶(hù).Session beans 也能用于事務(wù),它能夠更新共享的數據,但它不直接描繪這些共享的數據。Session beans 的生命周期是相對較短的。典型的是,只有當用戶(hù)保持會(huì )話(huà)的時(shí)候,Session beans 才是活著(zhù)的。一旦用戶(hù)退出了,Session beans 就不再與用戶(hù)相聯(lián)系了。Session beans被看成是瞬時(shí)的,因為如果容器崩潰了,那么用戶(hù)必須重新建立一個(gè)新的Session對象來(lái)繼續會(huì )話(huà)。 Session bean典型的聲明了與用戶(hù)的互操作或者會(huì )話(huà)。也就是說(shuō),Session bean了在客戶(hù)會(huì )話(huà)期間,通過(guò)方法的調用,掌握用戶(hù)的信息。一個(gè)具有狀態(tài)的Session bean稱(chēng)為有狀態(tài)的Session bean.當用戶(hù)終止與Session beans互操作的時(shí)候.會(huì )話(huà)終止了,而且,bean 也不再擁有狀態(tài)值。Session bean也可能是一個(gè)無(wú)狀態(tài)的 session bean.無(wú)狀態(tài)的Session beans并不掌握它的客戶(hù)的信息或者狀態(tài)。用戶(hù)能夠調用beans的方法來(lái)完成一些操作。但是,beans只是在方法調用的時(shí)候才知道用戶(hù)的參數變量。當方法調用完成以后,beans并不繼續保持這些參數變量。這樣,所有的無(wú)狀態(tài)的session beans的實(shí)例都是相同的,除非它正在方法調用期間。這樣,無(wú)狀態(tài)的Session beans就能夠支持多個(gè)用戶(hù).容器能夠聲明一個(gè)無(wú)狀態(tài)的Session beans.能夠將任何Session beans指定給任何用戶(hù). Entity Beans對數據庫中的數據提供了一種對象的視圖。例如:一個(gè)Entity bean能夠模擬數據庫表中一行相關(guān)的數據。多個(gè)client能夠共享訪(fǎng)問(wèn)同一個(gè)Entity bean.多個(gè)client也能夠同時(shí)的訪(fǎng)問(wèn)同一個(gè)Entity bean.Entity beans通過(guò)事務(wù)的上下文來(lái)訪(fǎng)問(wèn)或更新下層的數據。這樣,數據的完整性就能夠被保證。Entity Beans能存活相對教長(cháng)的時(shí)間,并且狀態(tài)是持續的。只要數據庫中的數據存在,Entity beans就一直存活。而不是按照應用程序或者服務(wù)進(jìn)程來(lái)說(shuō)的。即使EJB容器崩潰了,Entity beans也是存活的。Entity Beans生命周期能夠被容器或者 Beans自己管理。如果由容器控制著(zhù)保證 Entity beans持續的issus。如果由Beans自己管理,就必須寫(xiě)Entity beans的代碼,包括訪(fǎng)問(wèn)數據庫的調用。 Entity Beans是由主鍵(primary key 一種唯一的對象標識符)標識的。通常,主鍵與標識數據庫中的一塊數據,例如一個(gè)表中的一行,的主鍵是相同的。主鍵是client能夠定位特定的數據塊。 五、EJB的編程環(huán)境: 1、 使用Jbuilder Jbuilder與EJB Container能夠進(jìn)行無(wú)縫連接。Jbuilder和Inprise的應用服務(wù)器包括了所有的開(kāi)發(fā)和配置Enterprise Beans的工具以及所需要的庫:運行和管理Enterprise Bean的容器、命名服務(wù)、 事務(wù)服務(wù)、Java數據庫、開(kāi)發(fā)Enterprise Beans所需要的API、一個(gè)增強的java-to-iiop編譯器,支持值類(lèi)型和RMI信號等等。 Jbuilder還提供了一個(gè)快速開(kāi)發(fā)應用程序Enterprise Beans的工具和向導。通過(guò)簡(jiǎn)單而且直觀(guān)的步驟,向導幫助你建立一個(gè)Enterprise Bean。自己設定某些缺省值,產(chǎn)生了bean的模板,在上面,我們可以增加我們自己的應用邏輯。Jbuilder也提供了一個(gè)EJB的接口生成向導。向導在Enterprise Bean的公共方法基礎上生成了Remote接口和Home接口。Jbuilder還提供一個(gè)配置器的向導幫助我們逐步的建立XML描述器文件。并將生成的Stubs集中到一個(gè)jar文件中。 2、使用Jbuilder之外的集成環(huán)境: 如果你使用其它的除了別的集成環(huán)境(IDE)。要確定使用了集成環(huán)境IDE所帶的容器工具。也要驗證IDE是否支持EJB規范的相應的版本,還要確定它是否正確的支持EJB的API。 要確定JD到所支持的EJB容器的版本??梢酝ㄟ^(guò)檢查Inprise的安裝說(shuō)明來(lái)確定EJB容器所支持的支持JDK的版本。 在配置Enterprise Bean的時(shí)候,你必須使用Inprise的應用服務(wù)器所提供的工具。這些工具能夠編輯和修改第三方的代理商提供的Inprise配置描述器。還能夠驗證配置描述器,能夠驗證bean的源代碼。 六、一個(gè)簡(jiǎn)單的HELLO例子 1、安裝Apusic Application Server Note:以下以L(fǎng)inux為例,來(lái)說(shuō)明Apusic Application Server的安裝過(guò)程。其他平臺的安裝,可參考Apusic Application Server安裝手冊。 下載JDK1.2,Apusic Application Server必須運行在JDK1.2以上環(huán)境中??蓮囊韵抡军c(diǎn)下載最新JDK。 http://java.sun.com 下載Apusic Application Server Apusic Application Server 試用版可從以下網(wǎng)址得到: http://www.apusic.com/download/enter.jsp 在下載完成后,你可以得到一個(gè)包裹文件apusic.zip,選定安裝目錄,假設安裝到/usr下,則用以下命令: cd /usr jar xvf apusic.zip /usr下會(huì )出現一個(gè)目錄apusic,Apusic Application Server的所有程序都被解壓到/usr/apusic下。 將以下路徑加入到CLASSPATH中 /usr/apusic/lib/apusic.jar $JAVA_HOME/lib/tools.jar 用以下命令運行Apusic Application Server java -Xms64m com.apusic.server.Main -root /usr/apusic 2、定義EJB遠程接口(Remote Interface) 任何一個(gè)EJB都是通過(guò)Remote Interface被調用,EJB開(kāi)發(fā)者首先要在Remote Interface中定義這個(gè)EJB可以被外界調用的所有方法。執行Remote Interface的類(lèi)由EJB生成工具生成。 以下是HelloBean的Remote Inteface程序: package ejb.hello; import java.rmi.RemoteException; import java.rmi.Remote; import javax.ejb.*; public interface Hello extends EJBObject, Remote { //this method just get "Hello World" from HelloBean. public String getHello() throws RemoteException; } 3、定義Home Interface EJB容器通過(guò)EJB的Home Interface來(lái)創(chuàng )建EJB實(shí)例,和Remote Interface一樣,執行Home Interface的類(lèi)由EJB生成工具生成。 以下是HelloBean 的Home Interface程序: package ejb.hello; import javax.ejb.*; import java.rmi.Remote; import java.rmi.RemoteException; import java.util.*; /** * This interface is extremely simple it declares only * one create method. */ public interface HelloHome extends EJBHome { public Hello create() throws CreateException, RemoteException; } 4、寫(xiě)EJB類(lèi) 在EJB類(lèi)中,編程者必須給出在Remote Interface中定義的遠程方法的具體實(shí)現。EJB類(lèi)中還包括一些 EJB規范中定義的必須實(shí)現的方法,這些方法都有比較統一的實(shí)現模版,編程者只需花費精力在具體業(yè)務(wù)方法的實(shí)現上。 以下是HelloBean的代碼: package ejb.hello; import javax.ejb.*; import java.util.*; import java.rmi.*; public class HelloBean implements SessionBean { static final boolean verbose = true; private transient SessionContext ctx; // Implement the methods in the SessionBean // interface public void ejbActivate() { if (verbose) System.out.println("ejbActivate called"); } public void ejbRemove() { if (verbose) System.out.println("ejbRemove called"); } public void ejbPassivate() { if (verbose) System.out.println("ejbPassivate called"); } /** * Sets the session context. * * @param SessionContext */ public void setSessionContext(SessionContext ctx) { if (verbose) System.out.println("setSessionContext called"); this.ctx = ctx; } /** * This method corresponds to the create method in * the home interface HelloHome.java. * The parameter sets of the two methods are * identical. When the client calls * HelloHome.create(), the container allocates an * instance of the EJBean and calls ejbCreate(). */ public void ejbCreate () { if (verbose) System.out.println("ejbCreate called"); } /** * **** HERE IS THE BUSINESS LOGIC ***** * the getHello just return a "Hello World" string. */ public String getHello() throws RemoteException { return("Hello World"); } } 5、創(chuàng )建ejb-jar.xml文件 ejb-jar.xml文件是EJB的部署描述文件,包含EJB的各種配置信息,如是有狀態(tài)Bean(Stateful Bean) 還是無(wú)狀態(tài)Bean(Stateless Bean),交易類(lèi)型等。ejb-jar.xml文件的詳細信息請參閱EJB規范。以下是HelloBean的配置文件: <?xml version="1.0"?> <!DOCTYPE ejb-jar PUBLIC "-//Sun Microsystems Inc.//DTD Enterprise JavaBeans 1.2//EN" "http://java.sun.com/j2ee/dtds/ejb-jar_1_2.dtd"> <ejb-jar> <enterprise-beans> <session> <ejb-name>Hello</ejb-name> <home>ejb.hello.HelloHome</home> <remote>ejb.hello.Hello</remote> <ejb-class>ejb.hello.HelloBean</ejb-class> <session-type>Stateless</session-type> <transaction-type>Container</transaction-type> </session> </enterprise-beans> <assembly-descriptor> <container-transaction> <method> <ejb-name>Hello</ejb-name> <method-name>*</method-name> </method> <trans-attribute>Required</trans-attribute> </container-transaction> </assembly-descriptor> </ejb-jar> 6、編譯和部署 編譯Java源文件并將編譯后class和ejb-jar.xml打包到Hello.jar mkdir build mkdir build/META-INF cp ejb-jar.xml build/META-INF javac -d build *.java cd build jar cvf Hello.jar META-INF ejb cd .. 用EJB工具生成可部署到Apusic Application Server中運行的jar文件: java com.apusic.ejb.utils.EJBGen -d /usr/apusic/classes/Hello.jar build/Hello.jar 增加/usr/apusic/classes/Hello.jar到CLASSPATH中 將Hello.jar加入到Apusic Application Server配置文件中。在/usr/apusic/config/server.xml 加入以下幾行: <module> <ejb> <ejb-uri>classes/Hello.jar</ejb-uri> <bean> <ejb-name>Hello</ejb-name> <jndi-name>HelloHome</jndi-name> </bean> </ejb> </module> 啟動(dòng)服務(wù)器 java -Xms64m com.apusic.server.Main -root /usr/apusic 7、寫(xiě)客戶(hù)端調用程序 您可以從Java Client,JSP,Servlet或別的EJB調用HelloBean。 調用EJB有以下幾個(gè)步驟: 通過(guò)JNDI(Java Naming Directory Interface)得到EJB Home Interface 通過(guò)EJB Home Interface 創(chuàng )建EJB對象,并得到其Remote Interface 通過(guò)Remote Interface調用EJB方法 以下是一個(gè)從Java Client中調用HelloBean的例子: package ejb.hello; import javax.naming.Context; import javax.naming.InitialContext; import java.util.Hashtable; import javax.ejb.*; import java.rmi.RemoteException; /** * @author Copyright (c) 2000 by Apusic, Inc. All Rights Reserved. */ public class HelloClient{ public static void main(String args[]){ String url = "rmi://localhost:6888"; Context initCtx = null; HelloHome hellohome = null; try{ Hashtable env = new Hashtable(); env.put(Context.INITIAL_CONTEXT_FACTORY, "com.apusic.jndi.InitialContextFactory"); env.put(Context.PROVIDER_URL, url); initCtx = new InitialContext(env); }catch(Exception e){ System.out.println("Cannot get initial context: " + e.getMessage()); System.exit(1); } try{ hellohome = (HelloHome)initCtx.lookup("HelloHome"); Hello hello = hellohome.create(); String s = hello.getHello(); System.out.println(s); }catch(Exception e){ System.out.println(e.getMessage()); System.exit(1); } } } 運行HelloClient,可得到以下輸出: Hello World |
聯(lián)系客服