EJB中的三中bean:
会话Bean(Session Bean)
负责与客户端交互,是编写业务逻辑的地方,在会话bean中可以通过JDBC直接操作数据库,但大多数情况下都是通过实体bean来完成对数据库的操作。
实体Bean(Entity Bean)
它实际上属于java持久化规范(JPA)里的技术,JPA的出现主要是为了简化现有的持久化开发工作和整合ORM技术,结束现在Hibernate、TopLink等ORM框架各自为营的局面。
消息驱动Bean(MessageDriven Bean)
它是专门用于异步处理Java消息的组件。具有处理大量并发消息的能力。
会话bean:
无状态会话bean
平常我们使用最多的是无状态bean,因为它的bean实例可供多个用户使用,所以它的性能比有状态bean高。正因为一个bean实例被多个用户使用,那么前一个用户设置的值可能被后一个用户所修改,所以它无法正确保存某个用户设置的值,因此是无状态的。
有状态会话bean
有状态会话bean平常使用的并不多,因为它的一个bean实例只供一个用户使用,所以性能开销比较大,正因为它的实例只被一个用户使用,那么用户设置的值是不会被其他用户所修改的,所以可以正确保存用户设置的值,因此是有状态的。
开发工具:MyEclipse。
开发EJB依赖的jar文件:Jboss安装路径下的client下,通常把所以的jar添加到项目。
结构:
接口(可以是远程接口或本地接口)
实现类
步骤:
1.新建一个JavaProject:HelloWorld
2.导入所有的client的jar包
3.新建一个接口HelloWorld 在cn.hqu.ejb3下
添加方法 String sayHello(String name);
4.创建实现类:
新建一个类HelloWorldBean,实现HelloWorld接口,在包 cn.hqu.ejb3.impl;
实现方法里面:
return name+"说:你好!";
5.把这两个类变成ejb3
在HelloWorldBean加入注解 @Stateless
指定这个ejb是一个无状态会话bean;
指定这个接口是一个远程接口:(默认是本地接口)
在HelloWorldBean加入注解@Remote(HelloWorld.class) 。
Ejb开发完成
@Stateless
@Remote(HelloWorld.class)
public class HelloWorldBean implements HelloWorld {
@Override
public String sayHello(String name) {
// TODO Auto-generated method stub
return name+"说:你好!";
}
}
6.进行发布:
打成jar包,可以通过eclipse打包或者通过ant打包。
这里使用MyEclipse,右击项目选择Export→jar file 。
发布:先启动JBoss,复制打包好的jar拷贝到JBoss的server/default/deploy下,这样就完成ejb的部署;这样就发布成功了;会话bean开发完了。
Ejb在软件分层结构上,属于业务层,通常要被客户端调用,这里用Java类做ejb客户端。
新建类EJBClient 在包cn.hqu.test,在main里面编写EJB的调用代码:
package cn.hqu.test;
import java.util.Properties;
import javax.naming.InitialContext;
import javax.naming.NamingException;
import cn.hqu.ejb3.HelloWorld;
public class EJBClient {
/**
* @param args
*/
public static void main(String[] args) {
Properties pros = new Properties();
pros.setProperty("java.naming.factory.initial",
"org.jnp.interfaces.NamingContextFactory");
pros.setProperty("java.naming.provider.url", "localhost:1099");
try {
InitialContext ctx = new InitialContext(pros);
HelloWorld helloworld = (HelloWorld) ctx
.lookup("HelloWorldBean/remote");
System.out.println(helloworld.sayHello("苏志达"));
} catch (NamingException e) {
e.printStackTrace();
}
}
}
执行客户端(main方法);调用成功;
代码含义:客户端要调用ejb,要通过JNDI去寻找EJB的xx代理对象,JNDI是一套用来访问命名服务器的API,也是JavaEE里面的规范。
设置JNDI访问环境信息
如同数据库一样,根据访问命名服务器不同,为上下文设置的驱动类和URL也是不同的,如下面是访问Sun应用服务器的上下文信息:
如果客户端运行在应用服务器内,我们不需要为InitlalContext设置应用服务器的上下文信息,也不建议设置。因为应用服务器启动时会把JNDI驱动类等上下文信息添加进系统属性,创建InitialContext对象时如果没有指定Properties参数,InitialContext内部会调用System.getProperty()方法从系统属性里获得必要的上下文信息。对本例子而言,可以省略传入的props参数,在实际应用中如果给InitialContext设置了参数,反而会带来不可移植的问题。
注:创建InitialContext对象时如果没有指定Properties参数,InitialContext还会在classpath下寻找jndi.properties文件,并从该文件中加载应用服务器的上下文信息,这样避免了硬编码为InitialContext设置Properties参数。
InitialContext ctx = newInitialContext();
jndi.properties的配置如下:
java.naming.factory.initial=org.jnp.interfaces.NamingContextFactory
java.naming.provider.url=localhost\:1099
提示:这里编写的客户端程序完全可以部署在另外一台计算机上,只需要修改其中的连接服务器IP地址和启动Jboss时绑定相应IP地址即可。这正是EJB的远程调用特性。在第19行返回的helloworld实例对象并不是我们在服务器编写的HelloWorldBean,而是一个也实现了HelloWorld接口的代理对象,这个代理对象最终远程调用我们自己编写的HelloWorldBean,在19后加入如下代码,即可看到这个代理对象的类名:
System.out.println(helloworld.getClass().getName())
Jboss默认生成的JNDI名称
当EJB发布到Jboss时,如果我们没有为它指定全局JNDI名称或者修改过其默认EJB名称,Jboss就会按照默认的命名规则为EJB生成全局JNDI名称,默认是命名规则如下:
本地接口:EAR-FILE-BASE-NAME/EJB-CLASS-NAME/local
远程接口:EAR-FILE-BASE-NAME/EJB-CLASS-NAME/remote
EAR-FILE-BASE-NAME为ear文件的名称,EJB-CLASS-NAME为EJB的非限定类名。
例:把HelloWorld应用作为EJB模块打包进名为HelloWorld.ear的企业应用文件,它的远程接口的JNDI名称是:HelloWorld/HelloWorldBean/remote
如果把EJB应用打包成后缀为*.jar的模块文件,默认全局JNDI名称是
本地接口:EJB-CLASS-NAME/local
远程接口:EJB-CLASS-NAME/remote
例:把HelloWorld应用打包成HelloWorld.jar文件,它的远程接口的JNDI名称是:
HelloWorldBean/remote
代码:http://pan.baidu.com/s/1dDckCXN