设为首页 收藏本站
查看: 1246|回复: 0

[经验分享] EJB开发第二期---开发具有本地接口的无状态Bean

[复制链接]

尚未签到

发表于 2017-2-19 12:51:36 | 显示全部楼层 |阅读模式
一、EJB中的bean
1.1 EJB中bean分类
  会话bean(session bean)
  负责与客户端交互,是编写业务逻辑的地方,在会话bean中可以通过jdbc直接操作数据库,但大多数情况下都是通过实体bean来完成对数据库的操作。
  实体bean(entity bean)
  它实际上属于java持久化规范(简称JPA)里的技术,JPA的出现主要是为了简化现有的持久化开发工作和整合ORM技术,结束现在Hibernate、TopLink等ORM框架各自为营的局面。
  消息驱动bean(message-driven bean)
  它是专门用于异步处理java消息的组件,具有处理大量并发消息的能力。
1.2会话bean
  无状态会话bean
  平常,我们使用最多的是无状态bean,因为它的bean实例可供多个用户使用,所以它的性能比有状态bean高。正因为一个bean实例被多个用户使用,那么,前一个用户设置的值有可能被后一个用户所修改,所以它无法正确保存某个用户设置的值,因此是无状态的。
  有状态会话bean
  有状态bean平常使用的并不多,因为它的一个bean实例只供一个用户使用,所以性能开销比较大,正因为它的实例只被一个用户使用,用户为它设置的值是不会被其他用户修改,所以可以正确保存用户设置的值,因此是有状态的。
二、开发无状态会话bean
2.1 开发工具
  IDE工具:Eclipse Java EE IDE for Web Developers Version: Indigo Service Release 2
  JBoss服务器:jboss-4.2.3.GA
  JDK:JDK-1.6
  打包工具:Ant
  EJB依赖jar包:jboss安装路径的client目录下所有Jar文件以及javaee.jar
2.2 开发无状态会话bean
  在开发前,先熟悉一下无状态会话bean的调用流程图,如下图所示。
DSC0000.png

  01. 浏览器请求Test.jsp文件。
  02. 应用服务器的JSP引掣编译Test.jsp。
  03. Tast.jsp通过JNDI查找获得HelIoWorld EJB的存根对象,然后调用SayHello{)方法,EJB容器截获到方法调用。
  04. EJB容器调用HeIIoWorld实例的SayHello()方法 。
  05. 返回客户端浏览器。
2.3 开发步骤
  无状态会话bean的开发步骤如下:
  (1) 定义一个包含业务方法的接口
  这个接口不需要包含任何注释,它是一个普通的Java接口。调用EJB的客户端,使用这个接口引用从EJB容器返回的存根( stub)。代码如下:
  package ejb3Hello;
  public interface HelloWorld {
      public String SayHello(String name);
  }
  (2) 编写Bean class
  HeIIoWorldBean.java。Bean类推荐的命名方式是"接口+Bean",如HeIIoWorldBean。代码如下:
  package ejb3Hello.impl;
  import ejb3Hello.HelloWorld;
  import javax.ejb.Remote;
  import javax.ejb.Stateless;
  @Stateless
  @Remote({HelloWorld.class})  
  public class HelloWorldBean implements HelloWorld{
      @Override
      public String SayHello(String name) {         
          return name+"say:hello,this is my first EJB3.0.";
      }
  }
  在Bean类上面有两个注释@Stateless和@Remote,@Stateless注释指明这是一个无状态会话Bean。@Stateless注释的定义如下:
  Package javax.ejb;
  @Target(TYPE) @Retention(RUNTIME)
  public @interface Stateless {
   String name() default "";
   String mappedName() default "";
  }
  name()属性用于指定Session Bean的EJB名称。该名称在EJB Jar包中必须是全局唯一的,而在EAR中却可以重复,因为EAR可以包含多个EJB JAR,而每个JAR可以存在一个同名的EJB。在EAR中要定位某个EJB,可以这样使用:xxx.jar#HeIloWorldBean。如果不指定该属性,默认就是Bean class的非限定名称。对本例而言,EJB名称默认为HeIIoWorldBean。
  mappedName()属性指定Bean的全局JNDI名称,这个属性在WebLogic、Sun应用服务器和glassfish起作用。
  @Remote注释指定这个无状态Bean的remote接口。Bean类可以具有多个remote接口,每个接口之间用逗号分隔,如:@Remote({HeIIoWorld.class,Hello.class,World.class})。
  如果只有一个接口,则可以省略大括号,对于本例而言,可以写成这样:@Remote(HeIloWorld.class)。
  经过上面两步,一个HeIloWorld EJB就开发完了。现在将它发布到JBoss中。在发布前需要把它打成JAR色。打包JAR的方法有很多,如使用jar命令、集成开发工具或者Ant。下面介绍两种常用的打包方式:Eclipse打包向导和Ant打包。
三、EJB任务打包
3.1 Jar命令打包
  jar命令打包比较简单,进入要被打包的文件根目录中,比如被打包程序的目录结构如下:
  |---D:\webapp
  |---Test.jsp
  |--- WEB-INF
  |---web.xml
  可以进入到D:\webapp目录下,执行如下命令:
  jar cvf EJBTest.jar war *
  此命令将把Web应用的根目录下的所有文件打包成EJBTest.war文件,参数一:表示打包方式,参数二:表示打包后的文件名,参数三:表示文件类型。打包后的文件内容如下:
DSC0001.png

3.2 Eclipse打包
  步骤一:选择打包程序右键或单击Flile菜单,如下图所示。
DSC0002.png

  步骤二:选择Export选项,如下图所示,同时选择打包的类型:jar文件,填写文件路径及文件名。
DSC0003.png

  步骤三:选择next选项,如下图所示,选择默认设置就行。
DSC0004.png

  步骤四:选择next选项,如下图所示,如果程序中有main函数,选择Main类,最后选择finish。
DSC0005.png

3.3 Ant打包方式
  步骤一:选择项目右键,新建Ant的build.xml文件。
DSC0006.png

  步骤二:在 build.xml文件,可以编写所要做的工作,包括如下:
  01. 对应用进行编译
  02. 对应用进行打包
  03. 对应用进行发布
  04. 对应用进行解发
  build.xml文件内容如下:
  <?xml version="1.0" encoding="UTF-8"?>
  <project name="HelloWorld",basedir=".">
      <property name="src.dir" value="${basedir}\src"/>
      <property environment="env"/>
      <property name="jboss.home" value="${env.JBOSS_HOME}"/>
      <property name="jboss.server.config" value="default"/>
      <property name="build.dir" value="${basedir}\build"/>
      <path id="build.classpath">
          <fileset dir="${jboss.home}\client">
              <include name="*.jar"/>
          </fileset>
          <pathelement location="${build.dir}"/>
      </path>
      <target name="prepare">
          <delete dir="${build.dir}"/>
          <mkdir dir="${build.dir}" />
      </target>
      <target name="compile" depends="prepare" description="编译">
          <javac srcdir="${src.dir}" destdir="{build.dir}">
              <classpath refid="build.classpath"/>
          </javac>
      </target>
      <target name="ejbjar" depends="compile" description="创建EJB发布包">
          <jar jarfile="${basedir}\${ant.project.name}.jar">
              <fileset dir="${build.dir}">
                  <include name="**/*.class"
              </fileset>         
          </jar>
      </target>
  </project>
  这个打包任务建立了一个名为HelloWorld的Ant项目,项目的源目录由basedir属性来表示。该项目定义的四个属性如下:
  "src.dir":源文件路径
  "env":环境变量
  "jboss.home":Jboss安装目录
  "jboss.server.config":指定目前Jboss使用的配置项
  "build.dir":编译源文件的class类的目录
  接下来是一个类路径配置如下,它指定了应用程序依赖的jar文件,并且可以从配置项可以看出将编译存放的类路径也添加进来,如:pathelement。
  <path id="build.classpath">
  <fileset dir="${jboss.home}\client">
          <include name="*.jar"/>
      </fileset>
  <pathelement location="${build.dir}"/>
  </path>
  接下来又定义了一些具体任务,如下:
  <target name="prepare">
      <delete dir="${build.dir}"/>
      <mkdir dir="${build.dir}" />
  </target>
  该任务定义了,创建的build文件夹目录,用于存放编译后的jar文件,并定义了清空路径。
  <target name="compile" depends="prepare" description="编译">
      <javac srcdir="${src.dir}" destdir="{build.dir}">
          <classpath refid="build.classpath"/>
      </javac>
  </target>
  不难看出,该任务就是编译任务。通过javac命令对源文件进行编译。编译的源文件目录为:src.dir,编译后的源文件存放目录为:build.dir,编译过程中用的jar包通过refid来引用。并且该任务依赖于prepare任务,只有prepare任务先执行,该任务才可执行。
  <target name="ejbjar" depends="compile" description="创建EJB发布包">
      <jar jarfile="${basedir}\${ant.project.name}.jar">
          <fileset dir="${build.dir}">
              <include name="**/*.class"
          </fileset>         
      </jar>
  </target>
  该任务为打包任务,用到了jar命令对类文件进行打包。打包出来的路径:
  步骤三:在 build.xml文件,在Eclispe的Outline栏,执行build中的任务,首先执行编译。
DSC0007.png

  编译结果,Eclispe中console输出如下图所示:
DSC0008.png

  编译后的,项目结构如下:
DSC0009.png

  步骤四:在 build.xml文件,在Eclispe的Outline栏,执行打包,console端输出如下。
DSC00010.png

  打包后的项目结构如下:
DSC00011.png

  步骤五:在 build.xml文件,在Eclispe的Outline栏,执行发布,执行结果如下表示发布成功。
  19:16:45,605 INFO [AjpProtocol] Starting Coyote AJP/1.3 on ajp-127.0.0.1-8009
  19:16:45,611 INFO [Server] JBoss (MX MicroKernel) [4.2.3.GA (build: SVNTag=JBoss_4_2_3_GA date=200807181439)] Started in 9s:386ms
  19:17:45,945 INFO [JmxKernelAbstraction] creating wrapper delegate for: org.jboss.ejb3.stateless.StatelessContainer
  19:17:45,949 INFO [JmxKernelAbstraction] installing MBean: jboss.j2ee:jar=EJBTest.jar,name=HelloWorldBean,service=EJB3 with dependencies:
  19:17:45,975 INFO [EJBContainer] STARTED EJB: ejb3Hello.impl.HelloWorldBean ejbName: HelloWorldBean
  19:17:46,008 INFO [EJB3Deployer] Deployed: file:/F:/Tools/DevelopTool/javaserver/jboss-4.2.3.GA/server/default/deploy/EJBTest.jar
  网页查看方式如下:
DSC00012.png

DSC00013.png

DSC00014.png

  Global JNDI Namespace:
DSC00015.png

  如上图所示表示,EJB发布成功。 在上图所示页面中可以看到JBoss的JNDI树。其命名约定如下:
  (1) java:comp (Java:comp namespace)
  这个上下文环境和其子上下文环境仅能被应用组件内部访问和使用。
  (2) java: (Java: Namespace)
  子上下文环境和绑定的对象只能被处在同一个JVM内的客户访问。
  (3) Global JNDI Namespace
  上下文环境能被所有客户访问,不管它们是否处在同一个JVM内。
  当EJB发布到JBoss时,如果没有为它指定全局JNDI名称。JBoss就会按照默认的命名规则,为EJB生成全局JNDI名称。默认的命名规则如下:
  ■ 如果把EJB作为模块打包进后缀为*.ear的Java EE企业应用文件,默认的全局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模块打包进名为HeIIoWorld.ear的企业应用文件,它的远程接口的JNDI名称是HeIloWorld/HeIloWorldBean/remote。

  ■ 如果把EJB应用打包成后缀为*.j ar的模块文件,默认的全局JNDI名称如下。
  ■ 本地接口:EJB-CLASS-NAME/local
  ■ 远程接口:EJB-CLASS-NAME/remote
  :把HelIoWorld应用打包成HelIoWorld.jar文件,它的远程接口的JNDI名称是HelIoWorldBean/remote。
  注意:EJB-CLASS-NAME是不带包名的,如ejb3Hello.impl.HeIIoWorldBean只需取HeIIoWorldBean。在Glabal JNDI Namespace栏可以看到HeIIoWorldBean的远程接口的JNDI名称为,HeIIoWorldBean/remote,意味者EJB已经发布成功,接下来看看客户端如何访问它。
四、开发EJB客户端
4.1 EJB客户端代码
  public class EJBClient {
      public static void main(String[] args) {
          Properties props = new Properties();
          props.setProperty("java.naming.factory.initial","org.jnp.interfaces.NamingContextFactory");
          props.setProperty("java.naming.provider.url", "localhost:1099");
          try {
              InitialContext ctx = new InitialContext(props);
              HelloWorld helloworld = (HelloWorld) ctx.lookup("HelloWorldBean/remote");
              System.out.println(helloworld.sayHello("Sunddenly"));
          } catch (NamingException e) {
              System.out.println(e.getMessage());
          }
      }
  }
4.1 EJB客户端运行结果
  运行结果如下:
  Sunddenly say:hello,this is my first EJB3.0.
五、开发具有Local接口的Session bean
5.1 概述
  之前,我们介绍过远程接口。在这里,我们需要了解一下通过远程接口调用ejb的过程。首先客户端需要与ejb建立起socket通信,在通信管道上他们之间需要来回发送IIOP协议消息,因为数据要在网络进行传输,存放数据的java对象必须要进行序列化。
DSC00016.png

  在这个过程中我们看到,有网络通信的开销、协议解析的开销、对象序列化的开销。因为ejb是分布式技术,它允许客户端与ejb应用在不同一机器上面,所以这些性能开销也是必然的。但是在实际生产中,不可避免存在这种情况:客户端与EJB应用运行在同一个jboss中。这时候客户端访问ejb是否有必要走上面的网络通信呢?据我们所知,这时候客户端与ejb是在同一个jvm内,他们之间完全可以通过内存进行交互,这样就可以避免网络通信的性能开销。既然我们都想到了这一点,EJB专家组也想到了这一点,所以引入了本地接口。通过本地接口调用ejb,直接在内存中交互,这样就能避免因网络通信所造成的各种性能开销。但是有一点,大家必须注意,只有客户端与EJB应用在同一个JVM内运行的时候,我们才能调用本地接口,否则只能调用远程接口。谈到这里,有同学会问什么情况下客户端与EJB应用是在同一个JVM?简单地说只要客户端与ejb发布在同一个jboss内,我们就认为他们是在同一个JVM。
5.2 程序实现
  开发只有Local接口的无状态Session Bean的步骤和开发只有Remote接口的无状态会话Bean的步骤相同,两者唯一不同之处是,前者使用@Remote注释声明接口是远程接口,后者使用@Local注释声明接口是本地接口。当@Local和@Remote注释都不存在时,容器会将Bean class实现的接口默认为Local接口。如果EJB与客户端部署在同一个应用服务器。采用Local接口访问EJB优于Remote接口。因为通过Remote接口访问EJB需要在TCP/IP协议基础上转换和解释Corba IIOP协议消息,在调用EJB的这一过程中存在对象序列化,协议解释、TCP/IP通信等开销。而通过Local接口访问EJB是在内存中与Bean彼此交互的,没有了分布式对象协议的开销,大大改善了性能。下面是只有Loctd接口的无状态会话Bean。
  业务接口:LocaIHeIloWorld.java
  package ejb3Hello;
  public interface LocalHelloWorld {
      public String sayHello(String name);
  }
  Bean类:LocaIHeIloWorldBean.java
  package ejb3Hello.impl;
  import javax.ejb.Local;
  import javax.ejb.Remote;
  import javax.ejb.Stateless;
  import ejb3Hello.LocalHelloWorld;
  @Stateless
  @Local({LocalHelloWorld.class})
  public class LocalHelloWorldBean implements LocalHelloWorld{
      @Override
      public String sayHello(String name) {
          return name+"say:hello,this is my first EJB3.0.";
      }
  }
  @Local和@Remote注释一样,@Local注释也可以定义多个本地接口。如:@Local({ LocaIHelloWorld.class,Hello.class,World.class})。如果只有一个本地接口,可以省略大括号,对于本例而言,可以写成:@Remote(LocalHelIoWorld.class)。把上面的EJB打包成jar文件后,发布到"jboss安装目录/server/default/deploy目录中。接下来编写客户端调用代码。
  客户端类:
  将上面的代码打包成jar文件,发布到JBoss后会生成一个JNDI名称:beanname/lcoal。即,LocalHelloworldBean/local。下面按照Remote客户端方式编写的客户端代码如下:
  public class EJBClient {
      public static void main(String[] args) {
          try {
             InitialContext ctx = new InitialContext();
             LocalHelloWorld helloworld = (LocalHelloWorld) ctx.lookup("HelloWorldBean/local");
             System.out.println(helloworld.sayHello("Sunddenly "));
          } catch (NamingException e) {
              System.out.println(e.getMessage());
          }
      }
  }
  运行结果如下:
  Exception in thread "main" javax.ejb.EJBException: Invalid (i.e. remote) invocation of local interface (null container)
      at org.jboss.ejb3.stateless.StatelessLocalProxy.invoke(StatelessLocalProxy.java:80)
      at $Proxy0.sayHello(Unknown Source)
      at ejb3Hello.impl.test.EJBClient.main(EJBClient.java:15)
  运行过程中抛出了异常,大概意思为:无效的本地接口调用。这是因为目前,客户端和EJB应用在不同的JVM内,如果客户端和EJB应用在不同的JVM内中,我们只能通过远程接口调用EJB而不能通过本地接口调用EJB。那么,如果我们要通过本地接口调用EJB应用,就必须确保客户端和EJB在同一个JVM内,也就是说部署在同一个JBss中。
5.2 创建Web客户端应用
  考虑到大部分客户端应用都是Web应用,在这里我也建立一个Web应用作为EJB客户端,并把它部署到Jboos中。WEB应用创建过程如下:
  步骤一:创建Web工程,选择动态Web项目,如下:
DSC00017.png

  点击next,填写项目名称,和服务器类型
DSC00018.png

  next
DSC00019.png

  next,之后finish
DSC00020.png

  步骤二:在Web应用中创建Jsp文件,如下:
DSC00021.png

DSC00022.png

  next,之后finish
DSC00023.png

  步骤三:修改Web应用中创建的Jsp文件,格式改为"UTF-8",并把文件存放格式改为"UTF-8",如下:
  <%@ page language="java" contentType="text/html; charset=ISO-8859-1"
   pageEncoding="UTF-8"%>
  <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
  <html>
  <head>
  <meta http-equiv="Content-Type" content="text/html; charset=ISO-8859-1">
  <title>Insert title here</title>
  </head>
  <body>
  </body>
  </html>
DSC00024.png

DSC00025.png

  步骤四:编辑Test.jsp客户端,如下:
  <%@ page language="java" contentType="text/html; charset=ISO-8859-1"
   pageEncoding="UTF-8"%>
   <%@ page import="javax.naming.*,ejb3Hello.*" %>
  <html>
  <head>
  <meta http-equiv="Content-Type" content="text/html; charset=ISO-8859-1">
  <title>Insert title here</title>
  </head>
  <body>
  <%
  try {
      InitialContext ctx = new InitialContext();
      HelloWorld helloworld = (HelloWorld) ctx.lookup("HelloWorldBean/local");
      out.println(helloworld.sayHello("Local Person "));
  } catch (NamingException e) {
      out.println(e.getMessage());
  }
  %>
  </body>
  </html>
  步骤五:将程序Ant打包并发布,Ant 的build.xml文件如下:
  <?xml version="1.0" encoding="UTF-8"?>
  <project name="LocalHelloWorld" basedir=".">
      <property name="src.dir" value="${basedir}\src"/>
      <property environment="env"/>
      <property name="jboss.home" value="${env.JBOSS_HOME}"/>
      <property name="jboss.server.config" value="default"/>
      <property name="build.dir" value="${basedir}\build"/>
      <path id="build.classpath">
          <fileset dir="${jboss.home}\client">
              <include name="*.jar"/>
          </fileset>
          <pathelement location="${build.dir}"/>
      </path>
      <target name="prepare">
          <delete dir="${build.dir}"/>
          <mkdir dir="${build.dir}" />
      </target>
      <target name="compile" depends="prepare" description="编译">
          <javac srcdir="${src.dir}" destdir="${build.dir}">
              <classpath refid="build.classpath"/>
          </javac>
      </target>
      <target name="ejbjar" depends="compile" description="创建EJB发布包">
          <jar jarfile="${basedir}\${ant.project.name}.jar">
              <fileset dir="${build.dir}">
                  <include name="**/*.class"/>
              </fileset>         
          </jar>
      </target>
      <target    name="deploy" depends="ejbjar" description="发布ejb">
          <copy file="${basedir}\${ant.project.name}.jar" todir="${jboss.home}\server\${jboss.server.config}\deploy"/>
      </target>
      <target    name="undeploy" description="卸载ejb">
          <delete file="${jboss.home}\server\${jboss.server.config}\deploy\${ant.project.name}.jar"/>
      </target>
  </project>
5.3 创建Web客户端运行结果

  如果,您认为阅读这篇博客让您有些收获,不妨点击一下右下角的【推荐】。
如果,您希望更容易地发现我的新博客,不妨点击一下左下角的【关注我】。
如果,您对我的博客所讲述的内容有兴趣,请继续关注我的后续博客,我是【Sunddenly】。
  本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-344274-1-1.html 上篇帖子: 工欲善其事,必先利其器 下篇帖子: 性能工具列表
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表