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
|