站内搜索: 请输入搜索关键词
当前页面: 在线文档首页 > J2EE Tutorial 中文版

第4章 有状态会话Bean示例 - J2EE Tutorial 中文版


第4章 有状态会话Bean示例

会话Bean功能强大,因为它把客户端的区域扩展到了服务器(在服务器端保存客户端某些特定数据),然而它们仍然很容易创建。在第二章你已经建立了一个无状态会话Bean的例子ConvertEJB。这一章我们将创建一个购物车的有状态会话Bean CartEJB。

本章内容:

购物车会话Bean CartEJB

会话Bean类

Home接口

Remote接口

辅助类

运行该例子

其他企业Bean特征

访问环境实体

企业Bean的比较

传递企业Bean的对象引用

 

一.购物车会话Bean CartEJB

CartEJB描述一个在线书店的购物车。它的客户端的动作可能有:增加一本书,减少一本书,浏览购物车中存放的所有书。要建立该企业Bean,需要以下文件:

☆ 会话Bean类文件(CartBean.java)

☆ Home接口(CartHome.java)

☆ Remote接口(Cart.java)

所有的会话Bean都需要一个会话Bean类,所有允许远程访问的企业Bean都必须有一个Home接口和一个Remote接口(好像在哪说过了,不知道可不可以有多个:)。因为特定应用程序的需要,企业Bean可能也需要一些辅助类。本例的购物车用到两个辅助类:BookException和IdVerifier。这两个类将在后续节里介绍。

本例的原文件在j2eetutorial/examples/src/ejb/cart 目录下。要编译这些文件进入j2eetutorial/examples 目录执行如下命令:

ant cart

一个CartApp.ear的样本文件可以在j2eetutorial/examples/ears 目录下找到。

会话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工具中打开j2eetutorial/examples/ears/CartApp.ear 文件,你可以在图4-1中看到本例的CartApp应用程序

3. 部署CartApp应用程序(ToolsàDeploy)。在进入的对话框中确认你已经选择了Return Client JAR复选框

4. 运行

a) 在终端窗口中进入j2eetutorial/examples/ears 目录

b) 将环境变量APPCPATH设置为CartAppClient.jar文件的目录

c) 执行如下命令:

runclient -client CartApp.ear -name CartClient -textauth

d) 出现登录提示符后,输入用户名:guest,密码:guest123

图 4-1 CartApp 应用程序的General选项面板

二.其他的企业Bean特性

下面的这些特性是会话Bean和实体Bean的公有特性。

访问环境变量

在部署描述符中存储的环境变量是你不改变企业Bean的源代码也可以定制商业逻辑的名字-值对。例如一个计算商品折扣的企业Bean,用一个环境变量Discount Percent来存储折扣比例。在应用程序部署前,你可以用deploytool在Env Entries页里给Discount Percent赋值0.05。(如图4-2)当你运行该程序,企业Bean从它的运行环境中得到0.05的折扣比例。

Figure 4-2 CheckerBean的Env.Entries

在下面的示例代码中,applyDiscount方法根据购买数量用环境变量计算折扣。首先,它以java:comp/env参数调用lookup方法来定位环境命名上下文对象,然后用环境上下文对象的lookup方法得到Discount Level和Discount Percent环境变量的值。这里将把0.05赋给discountPercent变量。applyDiscount方法是CheckerBean类的方法,该类的源文件放在j2eetutorial/examples/src/ejb/checker 目录下,对应的样本CheckerApp.ear文件放在j2eetutorial/examples/ears目录下。

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());

   }

   ...