EJB3 阶段总结+一个EJB3案例 (2)

这篇博文接着上一篇博文的EJB案例。

在上一篇博文中,将程序的架构基本给描述出来了,EJB模块分为5层。

1)DB层,即数据库层

    在则一部分,我使用的数据库为mysql。在EJB程序中,访问数据库是通过Jboss中配置好的数据源进行的,然后在数据库中建立相应的数据库,不用建立表,在程序中使用JPA后通过Jboss启动会自动在数据库中间表

    具体的,可以看我之前写过的博文 wildfly8+jpa EntityBean 简单入门 在这篇博文中具体讲解了如何在Jboss中绑定一个数据源

2)dao层

    在这一层中,首先设计好具体的对象EntityBean,然后通过JPA的注解,将EntityBean和数据库进行映射。

    在这里还要使用一个EntityManage对象,该对象用于操作数据库,通过该对象可以使程序进行持久化,免去了JDBC时复杂的代码,该对象可分为容器管理和Bean管理,这里使用的是容器管理的EntityManage,通过@PersistenceContext注解将对象注入

    并且在该层中,通过接口的方式实现该层,并将该层定义为Local和Stateless,应为该层主要是用于该EJB的内部使用,是在一个JVM中的调用所以使用@Local注解的方式更加的节省资源。

    在该层中的方法均抛出继承RunTimeException的异常,为之后的Rollback做准备

3)Business层

    该层是业务层,其设计方式同dao层类似,均为实现类和接口方式,使用@Stateless和@Local将接口继续暴露给下一层。

    使用@EJB注解调用dao层的对象。(@EJB只能注入EJB对象,@Resource可以注入普通对象)

    在该层中,继续抛出RunTimeException异常,以支持回滚,在此处使用容器管理的事物方式,不需要代码管理开头结束,通过RunTimeException来管理

4)Service层

    服务层,主要是用于暴露给其他模块调用,在这里我使用的是WebService的方式,当然也可以通过Remote的方式(个人觉得太low),其设置为@Stateless的EJB

    在设计这一层代码时,应为使用的是WebService的模式,便产生一个问题:即现有Wsdl还是先有代码。前者的工作量无疑是大于后者的,在设计之初我也更倾向于后者,但项目组的技术大牛给了我很多的启示。一般的企业级javaee项目都会有许多不同的模块,甚至于在有的企业中还有C、C#等各种不同语言写的模块。而在这种情况下,如果使用的是先有代码再有wsdl,无疑开发速度将会加快,但在后期不同模块间使用WebService进行交互时会产生许多问题,如传输的参数在两个模块间类型不一致、参数的个数不一致等等。所以在这里我使用的是先有wsdl再有代码的方式。

     一个wsdl分为5个类型节点:types、message、porttype、binding、service。

     一个binding表示一个类。一个prottype中包含了具体的方法返回对象。message表示的是具体的方法参数通常成对出现。types中包含各种基本的类型(即组成方法参数的元素)。service表示的是服务,通常是将一个binding注册到服务上,service中包括了webservice访问的地址。

     为了便于维护,在一个webservice项目中,通常将types节点中的元素定义部分分割出去,即schema文件。(schema的作用类似与DTD,用于约束xml的节点元素,其适用性强于DTD,目前的项目大部分使用schema)

     话不多讲,先上wsdl和一个最基础的schema:

<?xml version="1.0" encoding="UTF-8" standalone="no"?>
<wsdl:definitions xmlns:soap="http://schemas.xmlsoap.org/wsdl/soap/"
    xmlns:tns="http://www.welv.com/wsdl/TeaStuService" xmlns:wsdl="http://schemas.xmlsoap.org/wsdl/"
    xmlns:xsd="http://www.w3.org/2001/XMLSchema" name="TeaStuService"
    targetNamespace="http://www.welv.com/wsdl/TeaStuService"
    xmlns:addGroupType="http://www.welv.com/schema/business/addGroupType"
    xmlns:addTeacheType="http://www.welv.com/schema/business/addTeacheType"
    xmlns:findTeacheType="http://www.welv.com/schema/business/findTeacheType">
    <wsdl:types xmlns:xsd="http://www.w3.org/2001/XMLSchema">
        <xsd:schema targetNamespace="http://www.welv.com/wsdl/TeaStuService">
            <xsd:import schemaLocation="../schema/business/addGroupType.xsd"
                namespace="http://www.welv.com/schema/business/addGroupType" />
            <xsd:import schemaLocation="../schema/business/addTeacheType.xsd"
                namespace="http://www.welv.com/schema/business/addTeacheType" />
            <xsd:import schemaLocation="../schema/business/findTeacheType.xsd"
                namespace="http://www.welv.com/schema/business/findTeacheType" />
        </xsd:schema>
    </wsdl:types>

    <wsdl:message name="addGroupRequest">
        <wsdl:part element="addGroupType:addGroupRequest" name="addGroupRequest" />
    </wsdl:message>
    <wsdl:message name="addGroupResponse">
        <wsdl:part element="addGroupType:addGroupResponse" name="addGroupResponse" />
    </wsdl:message>
    <wsdl:message name="addTeacheRequest">
        <wsdl:part element="addTeacheType:addTeacheRequest" name="addTeacheRequest" />
    </wsdl:message>
    <wsdl:message name="addTeacheResponse">
        <wsdl:part element="addTeacheType:addTeacheResponse" name="addTeacheResponse" />
    </wsdl:message>
    <wsdl:message name="findTeacheRequest">
        <wsdl:part element="findTeacheType:findTeacheRequest" name="findTeacheRequest" />
    </wsdl:message>
    <wsdl:message name="findTeacheResponse">
        <wsdl:part element="findTeacheType:findTeacheResponse" name="findTeacheResponse" />
    </wsdl:message>
    
    
    <wsdl:portType name="TeaStuService">
        <wsdl:operation name="addGroup">
            <wsdl:input message="tns:addGroupRequest" />
            <wsdl:output message="tns:addGroupResponse" />
        </wsdl:operation>
        <wsdl:operation name="addTeache">
            <wsdl:input message="tns:addTeacheRequest" />
            <wsdl:output message="tns:addTeacheResponse" />
        </wsdl:operation>
        <wsdl:operation name="findTeache">
            <wsdl:input message="tns:findTeacheRequest" />
            <wsdl:output message="tns:findTeacheResponse" />
        </wsdl:operation>
    </wsdl:portType>
    <wsdl:binding name="TeaStuServiceBinding-SOAP11HTTP" type="tns:TeaStuService">
        <soap:binding style="document"
            transport="http://schemas.xmlsoap.org/soap/http" />
        <wsdl:operation name="addGroup">
            <soap:operation soapAction="http://www.welv.com/wsdl/TeaStuService/addGroup" />
            <wsdl:input>
                <soap:body use="literal" />
            </wsdl:input>
            <wsdl:output>
                <soap:body use="literal" />
            </wsdl:output>
        </wsdl:operation>
        <wsdl:operation name="addTeache">
            <soap:operation soapAction="http://www.welv.com/wsdl/TeaStuService/addTeache" />
            <wsdl:input>
                <soap:body use="literal" />
            </wsdl:input>
            <wsdl:output>
                <soap:body use="literal" />
            </wsdl:output>
        </wsdl:operation>
        <wsdl:operation name="findTeache">
            <soap:operation soapAction="http://www.welv.com/wsdl/TeaStuService/findTeache" />
            <wsdl:input>
                <soap:body use="literal" />
            </wsdl:input>
            <wsdl:output>
                <soap:body use="literal" />
            </wsdl:output>
        </wsdl:operation>
    </wsdl:binding>
    <wsdl:service name="TeaStuService-SOAP11HTTP">
        <wsdl:port binding="tns:TeaStuServiceBinding-SOAP11HTTP"
            name="TeaStuServiceSOAP">
            <soap:address location="http://localhost:8080/service/TeaStuServiceSOAP" />
        </wsdl:port>
    </wsdl:service>
</wsdl:definitions>
wsdl
<?xml version="1.0" encoding="UTF-8"?>
<xsd:schema xmlns:xsd="http://www.w3.org/2001/XMLSchema"
    targetNamespace="http://www.welv.com/schema/common/common" xmlns:tns="http://www.welv.com/schema/common/common"
    elementFormDefault="qualified">

    <xsd:simpleType name="nameType">
        <xsd:restriction base="xsd:string"></xsd:restriction>
    </xsd:simpleType>

    <xsd:simpleType name="ageType">
        <xsd:restriction base="xsd:integer"></xsd:restriction>
    </xsd:simpleType>

    <xsd:simpleType name="addressType">
        <xsd:restriction base="xsd:string"></xsd:restriction>
    </xsd:simpleType>

    <xsd:simpleType name="idType">
        <xsd:restriction base="xsd:int"></xsd:restriction>
    </xsd:simpleType>


    <xsd:simpleType name="infoType">
        <xsd:restriction base="xsd:string"></xsd:restriction>
    </xsd:simpleType>


    <xsd:element name="name" type="tns:nameType"></xsd:element>

    <xsd:element name="address" type="tns:addressType"></xsd:element>

    <xsd:element name="age" type="tns:ageType"></xsd:element>
  
    <xsd:element name="id" type="tns:idType"></xsd:element>

    <xsd:element name="info" type="tns:infoType"></xsd:element>
</xsd:schema>
schema

     在schema中可以看到,在types节点中,使用了import节点,将schema引入了wsdl中,而schema则将方法的参数给细分,通过最基本的common文件来组合成各个方法的参数,有利于后期的维护,如Student中的name的字段长度设置为7,则可通过修改common文件中的定义便可。

     wsdl准备好后,将其放入EJB工程的Src目录下,然后使用命令:wsimport -keep -verbose -s E:ResourcesJavaEcAEJB-allejbModule E:ResourcesJavaEcAEJB-allejbModuleMETA-INFwsdlTeaStuService.wsdl   将wsdl对应的java代码生成到src目录下,这样一来wsdl对应的java代码便有了。

     以下是我使用命令生成的java代码,实现的带有Webservice功能的EJB代码:

@Stateless
@WebService(name = "TeaStuService", targetNamespace = "http://www.welv.com/wsdl/TeaStuService", wsdlLocation = "META-INF/wsdl/TeaStuService.wsdl")
public class TeaStuService_SOAP implements
        com.welv.wsdl.teastuservice.TeaStuService {

    @EJB
    private TeaStuService teaStuService;

    public AddGroupResponseType addGroup(AddGroupRequestType addGroupRequest) {
        // TODO Auto-generated method stub
        AddGroupResponseType dd = new AddGroupResponseType();
        dd.setInfo("wwwww");
        return dd;
    }

    public AddTeacheResponseType addTeache(AddTeacheRequestType addTeacheRequest) {

        TeacheType tt = addTeacheRequest.getTeache();
        Teache t = new Teache();
        t.setAddress(tt.getAddress());
        t.setAge(tt.getAge());
        t.setName(tt.getName());
        AddTeacheResponseType atr = new AddTeacheResponseType();
        atr.setInfo(teaStuService.addTeache(t));
        return atr;

    }

    public FindTeacheResponseType findTeache(
            FindTeacheRequestType findTeacheRequest) {
        Teache teache = teaStuService.findTeache(findTeacheRequest.getId());
        FindTeacheResponseType ftr = new FindTeacheResponseType();
        TeacheType tt = new TeacheType();
        tt.setAddress(teache.getAddress());
        tt.setAge(teache.getAge());
        tt.setId(teache.getId());
        tt.setName(teache.getName());
        ftr.setTeache(tt);
        return ftr;
    }

}

    该类是继承由wsdl生成的java接口的一个类。在类的开头使用@WebService注解,并将wsdl的相对路径注册其中,并要有name和targetnamespace两个属性

    将该类部署到Jboss8中,我使用的是Eclipse部署的方式。

   

    在控制台的log中可以看到一个地址,即wsdl中Service节点中定义的webservice地址,在该地址后面加上?wsdl即可访问部署到jboss上的webservice服务。

5)测试

    部署完EJB后,使用SoapUI进行测试,Soap的使用不在这里描述。

    使用webservice的访问地址将测试工程在SoapUI中建立好后,首先测试的是添加一个Teache对象:

    

     从图中的信息可看出访问成功,然后使用查找Teache的功能检验上一个case是否成功插入数据库

      

     查找成功,WebService和EJB的功能实现

*********************************************************

技术点:

1)wsdl开发中,使用的targetnamespace即目标名称空间,他表示的是文件本身,所以在文件中使用xmlns:tns=“targetnamespace URL”来表示应用自身的节点,而要使用其他的schema文件中的节点会使用import也是通过namespace来引用

2)在EJB中通常使用@Stateless的EJB,应为该类型占内存少

上述EJB的继续开发

1)目前使用的是SoapUI进行访问WebService服务,但实际的开发中,还是会有许多不同的模块,而由于历史原因这些模块的WebService接口往往是不同的,会导致不同模块间难以交互,这时可以使用ESB的技术,将ESB作为一个中转站,将不同的Wsdl的信息进行交换转发

2)在EJB的事物中,可以支持JDBC事物和JTA事物,我使用的是JTA事物,但在程序中只管理了JDBC的事物并没有跨资源管理事物

3)没有使用更加复杂的sql语句

4)可以使用Jmoke来进行单元测试

在上述的EJBdemo中,还有一些问题尚未解决

1)我之前设想的是将wsdl文件以及wsdl生成的java文件独立的使用一个java工程,然后将其打成jar包,在EJB中使用该jar包,但编译可以通过,部署到jboss中则会产生无法找到jar的异常(notfundclassException)

2)如何使用maven搭建一个EJB工程

3)事物管理的时候,runtimeException的作用不是很清楚

###代码下载

原文地址:https://www.cnblogs.com/welv/p/5146857.html