EJB应用开发的设计实例深入分析
EJB技术是Sun公司发明的,从发明至今已经发布了3个版本的EJB规范,1.0,1.1,2.0。EJB 2.1版已经发布了被提义的最终草案
EJB规范定义了组件开发者和EJB服务器厂商各自的职责。组件开发者根据EJB规范中定义的编程模型编写EJB。遵循EJB规范的EJB在各个EJB服务器中都可以移植。这样的一种编程模型给EJB开发者提供了很好的便利,EJB的产生背景是如何呢?
EJB技术将TP monitors和distributed-component services中的优点吸收过来,同时把它们的缺点抛弃掉。TP monitors是一种运行用过程语言,比如COBOL、PL/1,编写的应用程序可靠、功能强大的平台,该平台管理应用环境,比如事务性、安全性、资源管理、负载均衡、容错等方面。
所以这种平台优势给”mission-critical”企业级应用程序提供了很好的基础架构。同时,TP monitors采用业界先进的3层体系结构,即由表示层、商业逻辑层和后端资源组成。distributed-component services提供了更为先进的3层体系结构,即把商业对象、组件放置在中间层,其他的进程可以通过其提供的romote proxies来访问本身。
分布式组件和负荷在传统的TP monitors的基于过程的应用程序相比,其优点在于,它提供了更好的重用性和灵活性,因为它们可以将不同种类的商业对象集成起来,但这样的组件很难编写,同时缺少TP monitors所提供的健壮的基础框架。如果能够将两者的优势结合起来,便构成EJB出现的缘由。
EJB服务器给distributed components提供了类似TP monitors的环境。EJB服务器中,类似TP monitors环境的给出减少了开发的复杂度、distributed components的给出使得开发者的开发效率大大提高。
EJB客户通过JNDI、Java RMI-IIOP来访问。这种客户可能是Java应用程序、JSP(Servlets)、其他的EJB或者用其他语言写的应用。
设计实例的深入分析
要求:
本文假设读者对EJB技术有一定熟悉,对SQL有一定了解就可以了。由于EJB涉及到的技术很多,尽管每次只讲述一种技术,但各种技术是相关的,所以希望读者谅解,因为这样不是很好组织。
实例背景
数据库: SQL Server 2000
操作系统: Windows 2000
开发工具; JBuilder 7
EJB服务器: WebLogic 7.0
机器IP地址,10.11.12.58
实例综述:通过无状态Session Beans得到容器管理的Entity Beans中的数据,然后把无状态Session Beans得到的数据给JSP页面,从而客户可以看到所要的结果。通过这样一种过程使得您知道编写EJB组件是多么简单的事情。
准备工作
配置好SQL Server 2000的JDBC驱动(微软网站有下载,或者用BEA提供的也可以!该例子中用的是微软的JDBC)、JBuilder 7 + WebLogic 7集成环境,其他的也可以。
开发过程
1、配置JDBC数据源:这个过程包括两个步骤,首先配置连接池(Connection Pools)。(为什么采用连接池:开发人员不想编写Database方面的代码、更换数据库系统变得简单、限制数据库的连接数量、不需要为每个客户建立新连接。这种池的概念在J2EE中有很多地方出现了,比如EJB本身)配置的参数如下:
Name: cacd
URL: jdbc:microsoft:sqlserver:
//10.11.12.58:1433;user=sa;
password=cacd;
Driver ClassName:
com.microsoft.jdbc.sqlserver.SQLServerDriver
Initial Capacity: 3
Maximum Capacity: 10
其次,配置JDBC数据源,在Tx Data Sources中配置的参数如下:
Name: cacd
JNDI Name: cacd
Pool Name: cacd
(要与前面的相匹配!)
其中,在URL中,你需要为SQL Server 2000配置一个用户名sa,密码为cacd,因为这里用的是Type 4的JDBC,所以需要将SQL Server 2000的用户认证修改为:NT+SQL Server 2000混合认证。
2、数据库的建立:
Entity Beans代表了Database中的数据,所以需要数据库支持,但一般情况下,我们知道,可以根据容器管理的Entity Beans导出SQL DDL。另一方面,可以根据SQL DDL生成容器管理的Entity Beans。两种方法都可以。
比如JBuilder 7两种方法都提供了。但我想,信息模型(数据库)的建立在编码之前就应该存在,所以建议采用第二种办法。当然有些时候第一种会较为合理些,因为并不是表中所有的字段都会映射到Entity Beans中。该例子中建立了这样这样一个Table:
/*=============================*/
/* Table : techniquespec */
/*=============================*/
create table techniquespec
(
techniqueitem char(100) not null,
units char(10) null,
minvalue decimal(16,6) null,
maxvalue decimal(16,6) null,
types char(1) null,
signon char(1) null,
constraint PK_TECHNIQUESPEC
primary key (techniqueitem)
)
其中,开发人员在建表的过程中,不需要手工去写SQL DDL语句,一般都可以借助于工具进行,比如PowerDesigner、ERWin等工具。不要对工具产生不好的情绪,但前提是你熟悉数据库理论。用户建好Table后,可以填入数据,中文的也可以。
3、容器管理的Entity Beans的开发:JBuilder对EJB开发支持的比较好,提供了图形化的方式。由于我们已经建立好了Database,前面的techniquespec表,我们可以借助于Import Schema From Database,将SQL DLL引入进来。在这个过程中,一定要注意JNDI的名字和数据源中的JNDI要一致。
得到SQL DLL后,我们可以根据techniquespec表生成CMP 2.0 Entity Beans,在这里我们采用LocalHome访问Entity Beans,现在想返回表techniquespec中列techniqueitem的所有内容。
首先,真假一个findByTypes Finder方法,EJB QL语句为:
SELECT OBJECT(p) from Techniquespec AS p
其中返回值为Collection。下面给出代码。
LocalHome接口:
package cacdsystem;
import javax.ejb.*;
import java.util.*;
public interface TechniquespecHome
extends javax.ejb.EJBLocalHome
{
public Techniquespec create
(String techniqueitem) throws
CreateException;
public Collection findByTypes()
throws FinderException;
//添加的Finder方法
public Techniquespec
findByPrimaryKey(String techniqueitem)
throws FinderException;
}
Local接口:
package cacdsystem;
import javax.ejb.*;
import java.util.*;
import java.math.*;
public interface Techniquespec
extends javax.ejb.EJBLocalObject
{
public String getTechniqueitem();
public void setUnits(String units);
public String getUnits();
public void setMinvalue
(BigDecimal minvalue);
public BigDecimal getMinvalue();
public void setMaxvalue
(BigDecimal maxvalue);
public BigDecimal getMaxvalue();
public void setTypes(String types);
public String getTypes();
public void setSignon(String signon);
public String getSignon();
}
bean类:
package cacdsystem;
import javax.ejb.*;
abstract public class
TechniquespecBean implements EntityBean
{
EntityContext entityContext;
public java.lang.String ejbCreate
(java.lang.String techniqueitem)
throws CreateException
{
setTechniqueitem(techniqueitem);
return null;
}
public void ejbPostCreate
(java.lang.String techniqueitem)
throws CreateException
{
}
public void ejbRemove()
throws RemoveException
{
}
public abstract void setTechniqueitem
(java.lang.String techniqueitem);
public abstract void setUnits
(java.lang.String units);
public abstract void setMinvalue
(java.math.BigDecimal minvalue);
public abstract void setMaxvalue
(java.math.BigDecimal maxvalue);
public abstract void setTypes
(java.lang.String types);
public abstract void setSignon
(java.lang.String signon);
public abstract java.lang.String
getTechniqueitem();
public abstract java.lang.String
getUnits();
public abstract
java.math.BigDecimal getMinvalue();
public abstract
java.math.BigDecimal getMaxvalue();
public abstract
java.lang.String getTypes();
public abstract
java.lang.String getSignon();
public void ejbLoad()
{
}
public void ejbStore()
{
}
public