|
第4章 有状态会话Bean示例 会话Bean功能强大,因为它把客户端的区域扩展到了服务器(在服务器端保存客户端某些特定数据),然而它们仍然很容易创建。在第二章你已经建立了一个无状态会话Bean的例子ConvertEJB。这一章我们将创建一个购物车的有状态会话Bean CartEJB。 本章内容: 购物车会话Bean CartEJB 会话Bean类 Home接口 Remote接口 辅助类 运行该例子 其他企业Bean特征 访问环境实体 企业Bean的比较 传递企业Bean的对象引用
CartEJB描述一个在线书店的购物车。它的客户端的动作可能有:增加一本书,减少一本书,浏览购物车中存放的所有书。要建立该企业Bean,需要以下文件: ☆ 会话Bean类文件(CartBean.java) ☆ Home接口(CartHome.java) ☆ Remote接口(Cart.java) 所有的会话Bean都需要一个会话Bean类,所有允许远程访问的企业Bean都必须有一个Home接口和一个Remote接口(好像在哪说过了,不知道可不可以有多个:)。因为特定应用程序的需要,企业Bean可能也需要一些辅助类。本例的购物车用到两个辅助类:BookException和IdVerifier。这两个类将在后续节里介绍。 本例的原文件在
一个CartApp.ear的样本文件可以在j 会话Bean类 本例的会话Bean类是CartBean类。以CartBean为例,下面列出所有会话Bean都具有的特征: ☆ 它们都实现SessionBean接口(呵呵,它什么也不做,只是继承了Serializable接口而已) ☆ 会话Bean类必须声明为public ☆ 会话Bean类不能被定义为abstract或者final(就是说它不能是抽象类,而且允许被继承) ☆ 至少实现一个ejbCreate方法 ☆ 实现特定的商业方法 ☆ 包含一个无参数的构造函数 ☆ 不能定义finalize方法 以下是CartBean.java的代码: import java.util.*; import javax.ejb.*; public class CartBean implements 会话Bean { String customerName; String customerId; Vector contents; public void ejbCreate(String person) throws CreateException { if (person == null) { throw new CreateException("Null person not allowed."); } else { customerName = person; } customerId = "0"; contents = new Vector(); } public void ejbCreate(String person, String id) throws CreateException { if (person == null) { throw new CreateException("Null person not allowed."); } else { customerName = person; } IdVerifier idChecker = new IdVerifier(); if (idChecker.validate(id)) { customerId = id; } else { throw new CreateException("Invalid id: "+ id); } contents = new Vector(); } public void addBook(String title) { contents.addElement(title); } public void removeBook(String title) throws BookException { boolean result = contents.removeElement(title); if (result == false) { throw new BookException(title + "not in cart."); } } public Vector getContents() { return contents; } public CartBean() {} public void ejbRemove() {} public void ejbActivate() {} public void ejbPassivate() {} public void setSessionContext(SessionContext sc) {} } SessionBean接口 SessionBean接口继承EnterpriseBean接口,后者又继承Serializable接口。SessionBean接口声明了ejbRemove,ejbActivate,ejbPassivate和setSessionContext方法。虽然本例中CartBean没有用到这些方法,但也要实现它们,也为它们在SessionBean节口中声明过。因此在CartBean中被实现为空方法。大家可能都注意到了,该接口没有声明ejbCreate方法,该方法因为形式太自由(有状态会话Bean在实现的时候可能会有不确定个的参数,实体Bean的ejbCreate方法就更不用说了,不过可惜的是实体Bean并不实现该接口:),所以该方法无法作为固定类型的方法在接口中声明,该方法在被EJB容器调用时也只有以方法名作为标志,这可不太符合面向对象的原则。随后介绍什么时候使用这些方法。 ejbCreate方法 因为企业Bean运行在EJB容器中,所以客户端不可能直接创建一个企业Bean的实例,只有EJB容器有能力这么做。下面是本例创建CartEJB实例的过程: 1. 客户端调用Home对象的create方法 Cart shoppingCart = home.create("Duke DeEarl","123"); 2. EJB容器创建一个企业Bean(本例中是CartEJB)的实例 3. EJB容器调用企业Bean相应的ejbCreate方法,CartEJB中对应的方法: public void ejbCreate(String person, String id) throws CreateException { if (person == null) { throw new CreateException("Null person not allowed."); } else { customerName = person; } IdVerifier idChecker = new IdVerifier(); if (idChecker.validate(id)) { customerId = id; } else { throw new CreateException("Invalid id: "+ id); } contents = new Vector(); } 一般地,ejbCreate方法初始化企业Bean的状态。例如上述的ejbCreate方法就通过方法参数初始化customerName和customerId两个变量。 企业Bean必须至少实现一个ejbCreate方法(这句话好像重复了很多遍了,没办法这个确实很重要嘛:)。该方法签名必须符合以下要求: ☆ 方法的访问权修饰必须是public ☆ 返回值类型必须是void ☆ 如果该企业Bean允许远程访问,那么方法的参数必须是符合Java远程方法调用API(Java Remote Method Invocation,Java RMI)规则的合法类型,就是说必须是实现了Serializable接口或它的字接口的类的对象。 ☆ 方法不可以是static或final的 throws子句必须包含javax.ejb.CreateException异常。EjbCreate方法遇到非法的参数也会抛出该异常。 商业方法 会话Bean的主要作用就是执行客户端要求的商业逻辑。客户端调用Home对象的create方法返回的远程对象引用中的商业方法,在客户端透视图中,这些商业方法好像是在本地运行,实际上它们确实在远程会话Bean中运行。下面的代码片断展示了CartClient客户端如何调用商业方法: Cart shoppingCart = home.create("Duke DeEarl", "123"); ... shoppingCart.addBook("The Martian Chronicles"); shoppingCart.removeBook("Alice In Wonderland"); bookList = shoppingCart.getContents(); CartBean类中商业方法的实现如下 public void addBook(String title) { contents.addElement(new String(title)); } public void removeBook(String title) throws BookException { boolean result = contents.removeElement(title); if (result == false) { throw new BookException(title + "not in cart."); } } public Vector getContents() { return contents; } 商业方法必须遵循以下规则:l 方法名不能和EJB体系定义的方法冲突,例如不可以命名为ejbCreate或者ejbActivatel 访问权修饰符必须是public l 如果企业Bean允许远程访问,参数和返回值必须符合JavaRMI API调用规则 l 不可以用statci和final修饰符(和ejbCreate一样)throws子句可以包含任意类型的异常,不过要有意义。例如removeBook方法就会在购物车中没有要删除的书是抛出BookException异常。为了可以指出像无法连接数据库这样的系统错误,商业方法的应该在这时抛出javax.ejb.EJBException异常。当商业方法抛出该异常,容器会将其包装到一个RemoteException异常中,后者会被客户端捕获。容器不会擅自包装应用程序自定义异常,如本例中的BookException异常。因为EJBException异常是RutimeException(该异常类型系统会自动处理)异常类的子类,所以你不需要把它也加入到商业方法的throws子句中。Home接口企业Bean的Home接口继承javax.ejb.EJBHome接口。对于会话Bean,Home接口的目标是要定义客户端可访问create方法。如本例的CartClient客户端:Cart shoppingCart = home.create("Duke DeEarl", "123");Home接口中的每一个create方法都对应一个企业Bean类中的ejbCreate方法。CartBean的ejbCreate方法声明如下: public void ejbCreate(String person) throws CreateException ... public void ejbCreate(String person, String id) throws CreateException 对比一下CartHome接口中的create方法声明: import java.io.Serializable; import java.rmi.RemoteException; import javax.ejb.CreateException; import javax.ejb.EJBHome; public interface CartHome extends EJBHome { Cart create(String person) throws RemoteException, CreateException; Cart create(String person, String id) throws RemoteException, CreateException; } 以上两个文件中的ejbCreate和create方法声明很相似,但是也有重要的不同。下面是Home接口的create方法声明规则: ☆ 参数个数和类型必须和对应的ejbCreate方法一样,就是说要为企业Bean的每一个ejbCreate方法(永远不会用到的除外:)在Home接口中声明一个对应的参数个数类型和顺序一模一样的create方法 ☆ 参数和返回值类型必须符合RMI调用规则 ☆ create方法的返回值类型是企业Bean的Remote接口类型(对应的ejbCreate方法返回void) ☆ throws子句中必须包含java.rmi.RemoteException和javax.ejb.CreateException异常 Remote接口 Remote接口继承EJBObject接口,它定义客户端调用的商业方法。下面是本例中的Remote接口Cart.java的代码: import java.util.*; import javax.ejb.EJBObject; import java.rmi.RemoteException; public interface Cart extends EJBObject { public void addBook(String title) throws RemoteException; public void removeBook(String title) throws BookException, RemoteException; public Vector getContents() throws RemoteException; } Remote接口中的方法定义规则如下: ☆ 每一个商业方法声明必须在企业Bean类里实现有一个对应的商业方法 ☆ 必须与对应的企业Bean类实现的商业方法的声明签名一样,就是说参数类型个数顺序和返回值类型都必须一样。 ☆ 参数和返回值类型也必须符合RMI调用规则 ☆ throws子句必须包含RemoteException异常,就是在对应的企业Bean商业方法的throws子句的基础上加一个RemoteException组成Remote接口中相应方法的throws子句。 辅助类 本例有两个辅助类:BookException和IdVerifier。BookException异常类在removeBook方法找不到要删除的书时被抛出。IdVerifier在ejbCreate方法中被调用来验证custormerId的有效性。辅助类文件必须和调用它们的企业Bean类文件打包到同一个EJB JAR文件中。 运行本例 1. 启动J2EE服务器和deploytool工具 2. 在deploytool工具中打开 3. 部署CartApp应用程序(ToolsàDeploy)。在进入的对话框中确认你已经选择了Return Client JAR复选框 4. 运行 a) 在终端窗口中进入 b) 将环境变量APPCPATH设置为CartAppClient.jar文件的目录 c) 执行如下命令: runclient -client CartApp.ear -name CartClient -textauth d) 出现登录提示符后,输入用户名:guest,密码:guest123
图 4-1 下面的这些特性是会话Bean和实体Bean的公有特性。 访问环境变量 在部署描述符中存储的环境变量是你不改变企业Bean的源代码也可以定制商业逻辑的名字-值对。例如一个计算商品折扣的企业Bean,用一个环境变量Discount Percent来存储折扣比例。在应用程序部署前,你可以用deploytool在Env Entries页里给Discount Percent赋值0.05。(如图4-2)当你运行该程序,企业Bean从它的运行环境中得到0.05的折扣比例。
Figure 4-2 在下面的示例代码中,applyDiscount方法根据购买数量用环境变量计算折扣。首先,它以java:comp/env参数调用lookup方法来定位环境命名上下文对象,然后用环境上下文对象的lookup方法得到Discount
Level和Discount Percent环境变量的值。这里将把0.05赋给discountPercent变量。applyDiscount方法是CheckerBean类的方法,该类的源文件放在 public double applyDiscount(double amount) { try { double discount; Context initial = new InitialContext(); Context environment = (Context)initial.lookup("java:comp/env"); Double discountLevel = (Double)environment.lookup("Discount Level"); Double discountPercent = (Double)environment.lookup("Discount Percent"); if (amount >= discountLevel.doubleValue()) { discount = discountPercent.doubleValue(); } else { discount = 0.00; } return amount * (1.00 - discount); } catch (NamingException ex) { throw new EJBException("NamingException: "+ ex.getMessage()); } } 企业Bean的比较 客户端可以调用isIdentical方法来判断两个有状态会话Bean是否等价: bookCart = home.create("Bill Shakespeare"); videoCart = home.create("Lefty Lee"); ... if (bookCart.isIdentical(bookCart)) { // true ... } if (bookCart.isIdentical(videoCart)) { // false ... } 因为无状态会话Bean的对象都是等价的,所以用isIdentical方法比较他们时总是返回真。 实体Bean的等价判断也可以用isIdentical方法,不过这里还有另外一种方法,就是比较它们的主键: String key1 = (String)accta.getPrimaryKey(); String key2 = (String)acctb.getPrimaryKey(); if (key1.compareTo(key2) == 0) System.out.println("equal"); 访问企业Bean的远程对象引用 有时你的企业Bean需要提供一个自己的引用给其他企业Bean。例如你可以通过引用使企业Bean可以调用另一个企业Bean的方法。你无法访问this引用因为它指向在EJB容器中运行的Bean实例,只有容器可以直接调用Bean实例的方法。客户端通过远程接口实现对象间接调用Bean的方法,通过这些对象(远程接口实现对象)的引用企业Bean可以互相访问。 会话Bean调用SessionContext接口定义的getEJBObject方法获得它的远程接口对象引用。而实体Bean调用的是EntityContext接口定义的getEJBObject方法。这两个借口提供企业Bean访问EJB容器管理的上下文对象。典型情况下,企业Bean通过setSessionContext方法保存它的上下文对象。下面的代码片断说明了会话Bean如何使用这些方法: public class WagonBean implements SessionBean { SessionContext context; ... public void setSessionContext(SessionContext sc) { this.context = sc; } ... public void passItOn(Basket basket) { ... basket.copyItems(context.getEJBObject()); } ... |