• EJB开发第二期---开发具有本地接口的无状态Bean


    一、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的调用流程图,如下图所示。

    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文件,参数一:表示打包方式,参数二:表示打包后的文件名,参数三:表示文件类型。打包后的文件内容如下:

    3.2 Eclipse打包

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

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

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

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

    3.3 Ant打包方式

    步骤一:选择项目右键,新建Ant的build.xml文件。

    步骤二:在 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}uild"/>

        <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中的任务,首先执行编译。

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

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

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

    打包后的项目结构如下:

    步骤五:在 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

    网页查看方式如下:

    Global JNDI Namespace:

    如上图所示表示,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-NAMEear文件的名称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对象必须要进行序列化

    在这个过程中我们看到,有网络通信的开销协议解析的开销对象序列化的开销。因为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项目,如下:

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

    next

    next,之后finish

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

    next,之后finish

    步骤三:修改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>

    步骤四:编辑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}uild"/>

        <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】。

    本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。

  • 相关阅读:
    分布式架构高可用架构篇_activemq高可用集群(zookeeper+leveldb)安装、配置、高可用测试
    @interface [SpringMVC+redis]自定义aop注解实现控制器访问次数限制
    ActiveMQ安装与持久化消息
    activemq 5.13.2 jdbc 数据库持久化 异常 找不到驱动程序
    java通过Comparable接口实现字符串比较大小排序的简单实例
    微信小程序--火车票查询
    【调试】如何使用javascript的debugger命令进行调试(重要)
    【调试】js调试console.log使用总结图解(重要)
    ajax提交表单
    一个项目的404错误处理页面
  • 原文地址:https://www.cnblogs.com/sunddenly/p/4330246.html
Copyright © 2020-2023  润新知