在讲解如何注入其他EJB服务之前,我们首先先来看看什么是本地接口,第一个例子用的是远程接口,我们来介绍一下什么是本地接口。
1. 本地接口调用
之前我们讲解的远程调用接口的过程如下:
详解:首先客户端需要与ejb建立起socket通信,在通信管道上他们之间需要来回发送IIOP协议消息,因为数据要在网络进行传输,存放数据的java对象必须要进行序列化。这个过程有网络通信的开销,协议解析的开销(Internet Inter-ORB Protocol(互联网内部对象请求代理协议))原因在于ejb是分布式技术,它允许客户端与ejb应用在不同机器上面,所以这些性能开销是必然的。
但是实际的生活中我们不可能避免这种情况,那就是客户端与ejb应用运行在同一个jboss中,那么这个时候还是否有必要跟ejb之间走上面的网络通信呢?其实是不必要的,他们之间是完全可以通过内存进行交互的。这样就避免了上面的性能开销,故而引进了本地接口。通过本地接口调用ejb,直接通过内存进行交互。
但是需要注意一点:本地接口的调用,ejb和客户端的调用都在一个jvm内运行才可以的,才能调用本地接口,否则只能是调用远程接口,可能有人问,什么是在一个jvm呢运行呢?最简单的理解就是只要ejb和客户端发布在同一个jboss内,我们就认为是调用的本地接口。
此时将之前的接口改为local弹出,再次进行调用出现远程接口找不到。
开发一个web客户端
代码如下:
<%@ page language="java" import="java.util.*" pageEncoding="GB18030"%> <%@ page import="javax.naming.*,baidu.com.ejb.*" %> <% String path = request.getContextPath(); String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/"; %> <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"> <html> <head> <base href="<%=basePath%>"> <title>My JSP 'index.jsp' starting page</title> <meta http-equiv="pragma" content="no-cache"> <meta http-equiv="cache-control" content="no-cache"> <meta http-equiv="expires" content="0"> <meta http-equiv="keywords" content="keyword1,keyword2,keyword3"> <meta http-equiv="description" content="This is my page"> </head> <body> <%-- 其实java.lang.NoSuchMethodException这个异常的根本原因有两个:第一,缺少某些jar文件;第二, 某些jar文件有重复。--%> <%-- 之前总是报上面的错误,原因在于已经引进了ejb服务项目,在web端又进行了引用,故而弹出了这样的错误,删除引用即可--%> <%--并且在web端引用ejb服务--%> <%--因为没有书写<% %>导致输出的是源码输出,后台代码必须写上<% %>--%> <% try { InitialContext ctx = new InitialContext(); HelloWorld helloWorld = (HelloWorld) ctx .lookup("HelloWorldBean/local"); System.out.println(helloWorld.sayHello("小明")); } catch (NamingException e) { System.out.println(e.getMessage()); } %> </body> </html>
因为是在jsp页面书写代码而非html,需要用<%%>来编写后台的代码
2. 注解方式注入其他ejb
介绍完了本地接口的调用,那么下面我们开看看如何注入其他的ejb服务?
再声明一个接口和实现类为other
Other接口
package baidu.com.ejb; public interface Other { String sayMe(); }
OtherBean
package baidu.com.ejb.impl; import javax.ejb.Stateless; import baidu.com.ejb.Other; //默认是本地调用 @Stateless public class OtherBean implements Other { @Override public String sayMe() { return "Other"; } }
第一种是jndi查找
@Stateless @Local(HelloWorld.class) public class HelloWorldBean implements HelloWorld { // 这种方式是错误的,得到的不是ejb对象,而仅仅是一个java对象 // private Other other = new OtherBean(); // 第二种是ejb注入,实现的原理是:根据接口类型去寻找是否有实现了接口的ejb,寻找到结果的时候会把ejb注入进来 // 如果寻找到多个的时候,怎么办?明确指明调用的是哪一个即可。一种方式是修改实现bean的名字,使得变得唯一,另一种方式 @Override public String sayHello(String name) { // 第一种调用方式是:jndi查找 InitialContext ctx; try { ctx = new InitialContext(); Other other = (Other) ctx.lookup("OtherBean/local"); return name + "说:你好,世界!" + other.sayMe(); } catch (NamingException e) { e.printStackTrace(); } // InitialContext ctx = new InitialContext(); return name; } }
第二种是注解注入
@Stateless @Local(HelloWorld.class) public class HelloWorldBean implements HelloWorld { // 这种方式是错误的,得到的不是ejb对象,而仅仅是一个java对象 // private Other other = new OtherBean(); // 第二种是ejb注入,实现的原理是:根据接口类型去寻找是否有实现了接口的ejb,寻找到结果的时候会把ejb注入进来 // 如果寻找到多个的时候,怎么办?明确指明调用的是哪一个即可。一种方式是修改实现bean的名字,使得变得唯一,另一种方式 // 指明使用的哪一个即可(beanName="OtherBean") @EJB(beanName = "OtherBean") Other other; @Override public String sayHello(String name) { // 第一种调用方式是:jndi查找 try { return name + "说:你好,世界!" + other.sayMe(); } catch (NamingException e) { e.printStackTrace(); } return name; } }
此外我们还可以使用其他的注入,这个自己可以亲自实践。
3. 小结
讲到这里,ejb的讲解也算讲解了一部分了,在接下来的学习中会持续深入,更多地与大家分享和交流,还有一篇就是实体bean的讲解了,请持续关注~
版权声明:本文为博主原创文章,未经博主允许不得转载。