技術已經越來越多地應用到大型網絡系統開發中,本文中,筆者將介紹EJB(Enterprise Java Beans)的定義、基於EJB技術的應用系統結構模型以及EJB組件的內容和分類,最後結合基於EJB的結構模型和EJB組件開發了一個商務預訂系統。
EJB從技術上而言不是一種“產品”,而是一種技術規范。SUN公司對EJB的定義是:EJB的結構是開發和配置基於組件的分布式商務應用程序的一種組件結構。用EJB結構開發的應用程序是可伸縮的、事務型的、多用戶安全的。這些應用程序可能只需編寫一次,卻可以在支持EJB規范的任務服務器平台上進行配置。總的來說,EJB是一個組件事務監控的標准服務器端的組件模型。
基於EJB技術的系統結構模型
EJB結構是一個服務端組件結構,是一個層次性結構,其結構模型如圖1所示。該結構模型在通常情況下可分為客戶層、業務邏輯層和數據層,下面筆者對此作一簡單介紹。
圖2:商務預訂系統的構架
Entity Bean是為了現實世界的對象建造的模型,這些對象通常是數據庫的一些持久記錄。Entity Bean為那些可以表達成名詞的商務概念建立模型,它既描述了真實世界對象的狀態,也描述了它們的行為,同時允許開發者封裝與具體概念有關的數據和商務規則。Session Bean是一種通過Home Interface創建並對客戶端連接專有的Enterprise Bean,Session bean實例一般不與其它客戶端共享。Session Bean是客戶端應用程序的一個擴展,並負責管理整個過程或任務。Session Bean可以管理Entity Bean之間的交互,描述它們如何一起工作來完成一個特殊任務。
Entity Bean根據管理持久性的方式可以分為容器管理的Bean和使用Bean管理的Bean。容器管理的Bean由EJB容器自動管理它們的持久性,容器知道Bean實例的字段是怎樣映射到數據庫中去,並自動管理插入、更新和刪除數據庫中與實體有關的數據;使用Bean管理持久性的Bean需要明確地完成所有這些工作。Bean的開發者必須編寫代碼操作數據庫,EJB容器只告訴Bean實例什麼時候可以在數據庫中安全地插入、更新和刪除數據,除此之外,它不提供任何別的幫助。Bean實例自己完成所有的持久性工作。
Session Bean根據是否有狀態可分為無狀態Bean和有狀態Bean。無狀態Session Bean傾向於通用並可重復使用;有狀態Session Bean是客戶應用程序的擴展,它代表客戶完成任務並維護客戶的相關狀態。
商務預訂系統的開發
客艙預訂系統是在J2EE平台上開發的基於EJB組件技術的商務預訂系統,其主要流程是用戶登錄以後,將被依次帶領著通過顧客選擇頁和導航選擇頁,並將為顧客選擇一個可獲得的客艙(從TraveAgentBean處獲得可供選擇的客艙列表,TravelAgentBean的listAvailableCabin()方法由生成此網頁的Servlet調用,客艙列表將用於裝載到用戶浏覽器的網頁上創建HTML列表框),當用戶選擇一間客艙並提交了選擇時,一個HTTP請求將會發送到EJB服務器(Websphere Application Server),服務器接到此請求後,將其分派給ReservationServlet,此Servlet調用TravelAgent.BookPassage()方法做實際的預訂,BookPassage()方法返回的標簽信息將用於創建另一個送回用戶浏覽器的網頁。如果預訂成功,則由ProcessPaymentServlet去調用ProcessPaymentBean中的支付方法,從而實現對客戶的收費過程。其具體構架見圖2。
預定系統中的EJB組件主要包括以下幾個部分:
CabinBean:實體Bean,主鍵是CabinPK,是用來封裝現實世界中的航船客艙的一個實體Bean。
CustomerBean:實體Bean,主鍵是CustomerPK,是用來封裝現實世界中需要預訂航船客艙的消費者的一個實體Bean。
CruiseBean:實體Bean,主鍵是CruisePK,是用來封裝現實世界中航船航線的一個實體Bean。ReservationBean:實體Bean,主鍵是CruiseID,CabinID,它代表了數據庫中不變的一條記錄,即一個預訂,它記錄了預訂系統的歷史事件,主要是用來防止雙重預訂,即兩個客戶預訂相同航線的相同客艙,產生這種問題的原因是因為客戶選擇客艙和航線的時刻與調用bookPassage()方法的時刻之間有一段間隔時間。TravelAgentBean:有狀態會話Bean,一個負責預訂航行艙位工作流的會話Bean,它封裝了完成一條航線的一個預訂操作的過程並在全世界的旅行代理的客戶端應用程序中使用。TravelAgentBean不僅滿足消費者預訂票據的需要,還可提供在航行中剩余客艙的消息。為了完成此任務,Bean需要知道預訂是由哪一條航線、客艙,以及客戶組成,收集到這些信息後,由bookPassage()方法來完成處理預訂過程,它為客戶賬戶的計費負責,在正確航線的正確船只上預訂選擇的客艙,並通過Ticket類來為客戶產生一張票據。在這裡,我們需使用CreditCard類存儲有關客戶信用卡的相關信息,同時,ListAvailableCabins()方法用來顯示可用的尚未被預訂的客艙。
ProcessPaymentBean:無狀態會話Bean,它是在事務系統中向消費者收費的過程。它定義了支票、現金和信用卡支付方式的三個事務方法,即ByCheck()、ByCash()和ByCredit()。
商務預訂系統的程序代碼示例
整個商務預訂系統的開發是在IBM VisualAge for Java下面開發完成的,在該IDE開發環境中,實體Bean相對會話Bean要容易開發得多,以下就以TravelAgentBean為例,介紹EJB組件的開發過程:
1.TravelAgent遠程接口
它提供了設置客戶希望預訂的航線和客艙ID的方法。此外,還設置boolPassage()方法來對客戶的預訂進行計費,並為客戶產生一張票據。具體代碼如下:
package com.titan.travelagent;
import java.rmi.RemoteException;
import javax.ejb.FinderException;
import com.titan.cruise.Cruise;
import com.titan.customer.Customer;
import com.titan.processpayment.CreditCard;
public interface TravelAgent extends javax.ejb.EJBObject
{
public void setCruiseID(int cruise) throws RemoteException, FinderException;
public int getCruiseID( ) throws RemoteException, IncompleteConversationalState;
public void setCabinID(int cabin) throws RemoteException, FinderException;
public int getCabinID()throws RemoteException, IncompleteConversationalState;
public int getCustomerID()throws RemoteException, IncompleteConversationalState;
public Ticket boolPassage(CreditCard card,double price) throws RemoteException, IncompleteConversationalState;
}
2.TravelAgent Home接口
TravelAgent Home接口代碼如下:
puckage com.titan.tracelagent;
import java.rmi.RemoteException;
import javax.ejb.CreateException;
import com.titan.customer.Customer;
public interface TravelAgentHome extends javax.ejb.EJBHome {
public TravelAgent create(Customer cust) throws RemoteException,CreateException;}
3.TravelAgent Bean類
它需要實現TravelAgent的遠程接口和Home接口中的所有行為,限於篇幅,本文將不再介紹其實現代碼,感興趣的讀者可自己加以完成。
通過以上步驟,我們就完成了一個商務預定系統的EJB組件的開發。