spring创建web项目_使用Spring WS创建合同优先的Web服务

spring创建web项目

1引言

本文介绍了如何使用来实现和测试SOAP Web服务
Spring Web Services项目 。 本示例将JAXB2用于(取消)编组。 为了开发服务,我将使用合同优先的方法,该方法首先定义服务合同,然后基于此合同实施服务。

本文分为以下几节:

  • 2解释申请
  • 3实施服务
  • 3.1创建合同
  • 3.2生成Java类
  • 3.3实现SOAP端点
  • 3.4配置应用程序
  • 4测试服务
  • 5附加信息
  • 5.1实施客户
  • 5.2内部运作方式

2解释申请

示例应用程序处理订单。 我们有一个前端控制器(messageDispatcher servlet),它将处理订单请求,调用服务以处理订单并返回结果。

Aplicacio

您可以在github上获取源代码。

3实施服务

3.1创建合同

由于我们将使用“合同优先”方法,因此创建合同的最简单方法是首先定义示例xml文档,然后,我们将使用工具生成合同。 以下是示例xml文档:

client-request.xml

<clientDataRequest xmlns="http://www.xpadro.spring.samples.com/orders"clientId="A-123"productId="C5FH"quantity="5" />

client-response.xml

<clientDataResponse xmlns="http://www.xpadro.spring.samples.com/orders" confirmationId="7890B"orderDate="2013-09-22"amount="15.50" />

为了创建模式,我们可以使用Trang,这是一个开放源代码工具,将允许我们从xml文档中生成xsd模式。 我已经将该库包含到项目构建路径中(可以从Trang 网站获得此jar),并且创建了一个Ant任务来执行转换:

generate-schema.xml

<project name="Ant-Generate-Schema-With-Trang" default="generate" basedir="."><property name="src.dir" location="src" /><property name="trang.dir" location="lib" /><property name="source.dir" location="${src.dir}/main/webapp/WEB-INF/schemas/samples" /><property name="schema.dir" location="${src.dir}/main/webapp/WEB-INF/schemas/xsd" /><target name="generate" description="generates order schema"><delete dir="${schema.dir}" /><mkdir dir="${schema.dir}" /><java jar="${trang.dir}/trang.jar" fork="true"><arg value="${source.dir}/client-request.xml" /><arg value="${schema.dir}/client-request.xsd" /></java><java jar="${trang.dir}/trang.jar" fork="true"><arg value="${source.dir}/client-response.xml" /><arg value="${schema.dir}/client-response.xsd" /></java></target>
</project>

一旦执行了Ant任务,它将生成架构。 由于模式是自动生成的,因此我们可能需要进行一些修改以使其适应我们的需求。 让我们来看看:

客户端请求

<xs:schema xmlns:xs="http://www.w3.org/2001/XMLSchema" elementFormDefault="qualified" targetNamespace="http://www.xpadro.spring.samples.com/orders" xmlns:orders="http://www.xpadro.spring.samples.com/orders"><xs:element name="clientDataRequest"><xs:complexType><xs:attribute name="clientId" use="required" type="xs:NCName"/><xs:attribute name="productId" use="required" type="xs:NCName"/><xs:attribute name="quantity" use="required" type="xs:integer"/></xs:complexType></xs:element>
</xs:schema>

客户端响应

<xs:schema xmlns:xs="http://www.w3.org/2001/XMLSchema" elementFormDefault="qualified" targetNamespace="http://www.xpadro.spring.samples.com/orders" xmlns:orders="http://www.xpadro.spring.samples.com/orders"><xs:element name="clientDataResponse"><xs:complexType><xs:attribute name="amount" use="required" type="xs:decimal"/><xs:attribute name="confirmationId" use="required" type="xs:NMTOKEN"/><xs:attribute name="orderDate" use="required" type="xs:NMTOKEN"/></xs:complexType></xs:element>
</xs:schema>

我们可以向这些架构添加不同的验证,但是在本示例中,我将只修改几种类型,例如clientId,productId和ConfirmationId(xs:string)和orderDate(xs:date)。 XML数据类型到Java类型的映射由JAXB完成。 您可以检查提供了哪些映射
在这里 。

为了完成该架构,我们将把response元素复制到请求架构中。 我创建了带有响应和请求的第三个架构:

客户端服务

<xs:schema xmlns:xs="http://www.w3.org/2001/XMLSchema"elementFormDefault="qualified" targetNamespace="http://www.xpadro.spring.samples.com/orders"xmlns:orders="http://www.xpadro.spring.samples.com/orders"><xs:element name="clientDataRequest"><xs:complexType><xs:attribute name="clientId" use="required" type="xs:string" /><xs:attribute name="productId" use="required" type="xs:string" /><xs:attribute name="quantity" use="required" type="xs:integer" /></xs:complexType></xs:element><xs:element name="clientDataResponse"><xs:complexType><xs:attribute name="amount" use="required" type="xs:decimal" /><xs:attribute name="confirmationId" use="required" type="xs:string" /><xs:attribute name="orderDate" use="required" type="xs:date" /></xs:complexType></xs:element>
</xs:schema>

最后一步是写合同,通常用WSDL文件表示。 如果您不想手动创建它,Spring-ws项目为我们提供了一种从XSD模式生成此文件的方法。 您将在配置应用程序部分中看到第二种方法。

3.2生成Java类

我们将使用JAXB2生成请求和响应对象。 JAXB的XJC编译器将负责从我们之前生成的XSD架构转换这些对象。 它将作为Ant任务执行:

<project name="Ant-Generate-Classes-With-JAXB2" default="generate" basedir="."><property name="src.dir" location="src" /><property name="java.dir" location="src/main/java" /><property name="schema.dir" location="${src.dir}/main/webapp/WEB-INF/schemas/xsd" /><target name="generate"><exec executable="xjc"><arg line=" -d ${java.dir} -p xpadro.spring.ws.types ${schema.dir}/client-service.xsd" /></exec></target>
</project>

该任务将在xpadro.spring.ws.types包中创建Java类(您可能需要刷新项目)。

类型

3.3实现SOAP端点

端点接收未编组的消息有效负载,并使用此数据来调用订单服务。 然后它将返回服务响应,该响应将由端点适配器编组:

@Endpoint
public class OrderEndpoint {@Autowiredprivate OrderService orderService;@PayloadRoot(localPart="clientDataRequest", namespace="http://www.xpadro.spring.samples.com/orders")public @ResponsePayload ClientDataResponse order(@RequestPayload ClientDataRequest orderData) {OrderConfirmation confirmation = orderService.order(orderData.getClientId(), orderData.getProductId(), orderData.getQuantity().intValue());ClientDataResponse response = new ClientDataResponse();response.setConfirmationId(confirmation.getConfirmationId());BigDecimal amount = new BigDecimal(Float.toString(confirmation.getAmount()));response.setAmount(amount);response.setOrderDate(convertDate(confirmation.getOrderDate()));return response;}//date conversion
}

这是端点使用的注释的简短描述:

@Endpoint :将类注册为组件。 这样,将通过组件扫描检测到该类。

@PayloadRoot :将端点方法注册为请求的处理程序。 该注释将定义该方法可以处理的请求消息类型。 在我们的示例中,它将接收消息,其中其有效负载根元素具有与我们创建的XSD架构中定义的名称空间相同的名称空间,并且其本地名称是为请求定义的名称(clientDataRequest)。

@RequestPayload :指示要作为参数传递给方法的请求消息的有效负载。

@ResponsePayload ,指示将返回值用作响应消息的有效负载。

3.4配置应用程序

web.xml
应用程序配置(如数据源,transactionManager等)

<context-param><param-name>contextConfigLocation</param-name><param-value>classpath:xpadro/spring/ws/config/root-config.xml</param-value>
</context-param>

加载应用程序上下文

<listener><listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>

这是一个Servlet,它将充当处理所有SOAP调用的前端控制器。 它的功能是将传入的XML消息派生到端点,就像Spring MVC的DispatcherServlet一样。

<servlet><servlet-name>orders</servlet-name><servlet-class>org.springframework.ws.transport.http.MessageDispatcherServlet</servlet-class><init-param><param-name>contextConfigLocation</param-name><param-value>classpath:xpadro/spring/ws/config/servlet-config.xml</param-value></init-param><load-on-startup>1</load-on-startup>
</servlet><servlet-mapping><servlet-name>orders</servlet-name><url-pattern>/orders/*</url-pattern>
</servlet-mapping>

servlet-config.xml
此配置包含Web服务基础结构bean。

<!-- Detects @Endpoint since it is a specialization of @Component -->
<context:component-scan base-package="xpadro.spring.ws"/><!-- detects @PayloadRoot -->
<ws:annotation-driven/><ws:dynamic-wsdl id="orderDefinition" portTypeName="Orders" locationUri="http://localhost:8081/spring-ws"><ws:xsd location="/WEB-INF/schemas/xsd/client-service.xsd"/>
</ws:dynamic-wsdl>

在动态wsdl中,在locationUri属性中输入什么值都没有关系,因为它将由MessageDispatcherServlet处理。 因此,wsdl将在以下位置可用:

http:// localhost:8081 / spring-ws / orders / whatever / orderDefinition.wsdl

4测试服务

下面的示例创建一个模拟客户端,该客户端将访问Web服务:

@ContextConfiguration("classpath:xpadro/spring/ws/test/config/test-server-config.xml")
@RunWith(SpringJUnit4ClassRunner.class)
public class TestWebService {@AutowiredApplicationContext context;private MockWebServiceClient mockClient;@Testpublic void testValidOrderRequest() {Source requestPayload = new StringSource("<clientDataRequest xmlns='http://www.xpadro.spring.samples.com/orders' " +"clientId='123' productId='XA-55' quantity='5'/>");Source responsePayload = new StringSource("<clientDataResponse xmlns='http://www.xpadro.spring.samples.com/orders' " +"amount='55.99' confirmationId='GHKG34L' orderDate='2013-10-26+02:00'/>");RequestCreator creator = RequestCreators.withPayload(requestPayload);mockClient = MockWebServiceClient.createClient(context);mockClient.sendRequest(creator).andExpect(ResponseMatchers.payload(responsePayload));}@Testpublic void testInvalidOrderRequest() {Source requestPayload = new StringSource("<clientDataRequest xmlns='http://www.xpadro.spring.samples.com/orders' " +"clientId='456' productId='XA-55' quantity='5'/>");Source responsePayload = new StringSource("<SOAP-ENV:Fault xmlns:SOAP-ENV='http://schemas.xmlsoap.org/soap/envelope/'>" +"<faultcode>SOAP-ENV:Server</faultcode><faultstring xml:lang='en'>Client [456] not found</faultstring></SOAP-ENV:Fault>");RequestCreator creator = RequestCreators.withPayload(requestPayload);mockClient = MockWebServiceClient.createClient(context);mockClient.sendRequest(creator).andExpect(ResponseMatchers.payload(responsePayload));}
}

此测试中使用的配置文件非常简单,仅包含对服务组件的扫描:

<context:component-scan base-package="xpadro.spring.ws"/>
<ws:annotation-driven/>

5附加信息

5.1实施客户

为了方便客户端访问Web服务,Spring为我们提供了WebServiceTemplate类。 此类包含用于发送和接收消息的方法,并且还使用转换器对(取消)编组对象。

我创建了一个充当服务客户端的测试:

@ContextConfiguration("classpath:xpadro/spring/ws/test/config/client-config.xml")
@RunWith(SpringJUnit4ClassRunner.class)
public class TestClient {@Autowired WebServiceTemplate wsTemplate;@Testpublic void invokeOrderService() throws Exception {ClientDataRequest request = new ClientDataRequest();request.setClientId("123");request.setProductId("XA-55");request.setQuantity(new BigInteger("5", 10));ClientDataResponse response = (ClientDataResponse) wsTemplate.marshalSendAndReceive(request);assertNotNull(response);assertEquals(new BigDecimal("55.99"), response.getAmount());assertEquals("GHKG34L", response.getConfirmationId());}
}

配置测试文件包含WebServiceTemplate配置:

<oxm:jaxb2-marshaller id="marshaller" contextPath="xpadro.spring.ws.types"/><bean class="org.springframework.ws.client.core.WebServiceTemplate"><property name="marshaller" ref="marshaller" /><property name="unmarshaller" ref="marshaller" /><property name="defaultUri" value="http://localhost:8081/spring-ws/orders" /> 
</bean>

只需记住在执行此测试之前使用已部署的Web服务应用程序启动服务器即可。

5.2内部运作方式

如果您只想实现Web服务,那么上一节将结束本文。 对于那些对它如何真正起作用感到好奇的人,我将尝试解释如何将请求映射到端点,这比到目前为止的解释要低一些。

当请求到达MessageDispatcher时,它依赖于两个组件:

  1. 它询问EndpointMapping哪个是适当的端点。
  2. 利用从映射接收到的信息,它使用端点适配器来调用端点。 适配器还支持参数解析器和返回类型处理程序。

端点映射

MessageDispatcher包含一个端点映射列表,每个端点映射都包含一个先前注册的方法端点的映射。 在我们的例子中,JAXB映射PayloadRootAnnotationMethodEndpointMapping已注册所有带有@PayloadRoot注释的方法。 如果消息的有效负载的合格名称解析为注册方法,则它将被返回给MessageDispatcher。 如果我们不注释我们的方法,它将无法处理请求。
序列1
端点适配器

然后,MessageDispatcher将询问其每个端点适配器是否支持当前请求。 在我们的情况下,适配器检查以下两个条件是否都成立:

  • 传递给该方法的至少一个参数使用@RequestPayload进行注释
  • 如果端点方法返回响应,则必须使用@ResponsePayload进行注释

如果返回了适配器,则它将调用端点,在将参数传递给方法之前将其解组。 当该方法返回响应时,适配器将封送它。

下图是此步骤的简化版本,以简化操作:

序列2

结论

我们已经看到了有关如何实现简单的Web服务然后对其进行测试的介绍。 如果您有兴趣,还可以看看如何使用MockWebServiceServer 测试客户端。

参考: XavierPadró博客博客中的JCG合作伙伴 Xavier Padro 使用Spring WS创建了合同优先的Web服务 。

翻译自: https://www.javacodegeeks.com/2014/02/creating-contract-first-web-services-with-spring-ws.html

spring创建web项目

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/343893.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

什么是网络光端机?网络光端机工作原理及功能介绍!

网络光端机&#xff0c;就是光信号传输的终端设备。由于目前技术的提高&#xff0c;光纤价格的降低使它在各个领域得到很好的应用。在远程光纤传输中&#xff0c;光缆对信号的传输影响很小&#xff0c;光纤传输系统的传输质量主要取决于光端机的质量&#xff0c;因为光端机负责…

以太网交换机的用途有哪些?

交换机的主要功能包括物理编址、网络拓扑结构、错误校验、帧序列以及流控。交换机还具备了一些新的功能&#xff0c;如对VLAN&#xff08;虚拟局域网&#xff09;的支持、对链路汇聚的支持&#xff0c;甚至有的还具有防火墙的功能。接下来就由飞畅科技的小编带大家详细了解下交…

交换机主要分为哪几类?

交换机&#xff08;Switch&#xff09;意为“开关”是一种用于电&#xff08;光&#xff09;信号转发的网络设备。它可以为接入交换机的任意两个网络节点提供独享的电信号通路。最常见的交换机是以太网交换机。其他常见的还有电话语音交换机、光纤交换机等。 从广义上来看&…

java编程彩球滑梯作弊_课内资源 - 基于C++的学生成绩管理系统

1 系统分析1.1 学生成绩管理系统介绍班主任管家软件以学生信息、课程信息为基础&#xff0c;以品行表现成绩(辅导员、班主任、班级评议成绩)和业务课程成绩为评价依据&#xff0c;每学期评定一次奖学金&#xff0c;奖学金作为学生评定各种荣誉的主要依据&#xff0c;并将各种评…

交换机工作原理介绍

交换机是一种用于电&#xff08;光&#xff09;信号转发的网络设备。它可以为接入交换机的任意两个网络节点提供独享的电信号通路。最常见的交换机是以太网交换机。交换机工作于OSI参考模型的第二层&#xff0c;即数据链路层。交换机拥有一条高带宽的背部总线和内部交换矩阵&am…

bitmap的java原理_布隆算法的原理及JAVA实现

Bloom Filter是由Bloom在1970年提出的一种多哈希函数映射的快速查找算法。通常应用在一些需要快速判断某个元素是否属于集合&#xff0c;但是并不严格要求100%正确的场合。Bloom Filter是一种空间效率很高的随机数据结构&#xff0c;它利用位数组很简洁地表示一个集合&#xff…

什么是语音复用设备?

语音复用设备是一种小容量、多业务的复用及传输设备&#xff0c;它能为用户提供综合的通信功能&#xff0c;满足用户对电话、数据以及计算机网络连接的实际需要。它的多种传输方式使其能灵活的应用在各种通信网络中&#xff1a;可以作为光电一体化综合通信业务接入设备&#xf…

截断整型提升算数转换

文章目录 &#x1f680;前言&#x1f680;截断&#x1f680;整型提升✈️整型提升是怎样的 &#x1f680;算术转换 &#x1f680;前言 大家好啊&#xff01;这里阿辉补一下前面操作符遗漏的地方——截断、整型提升和算数转换 看这一篇要先会前面阿辉讲的数据的存储否则可能看不…

java如何通过grpc连接etcd_grpc通过 etcd 实现服务发现与注册-源码分析

介绍下面介绍 jupiter-0.2.7 版本中 grpc 通过 etcd 实现服务发现与注册。服务发现与注册的实现解析服务注册服务注册的流程图&#xff1a;etcd的服务注册代码模块在 jupiter/pkg/registry/etcdv3 中。下面让我们来看看实际的代码// Registry register/unregister service// re…

spring jms 事务_Spring JMS:处理事务中的消息

spring jms 事务1.引言 这篇文章将向您展示使用JMS异步接收消息期间使用者执行过程中的错误如何导致消息丢失。 然后&#xff0c;我将解释如何使用本地事务解决此问题。 您还将看到这种解决方案在某些情况下可能导致消息重复&#xff08;例如&#xff0c;当它将消息保存到数据…

使用AWS Lambda,S3和AWS CloudFront进行动态内容缓存

快速提供内容对于任何网站或应用程序具有更好的客户体验至关重要。 如果您将网站或应用程序托管在AWS Cloud中&#xff0c;那么无论从何处访问应用程序&#xff0c;都可以以较低的延迟快速提供内容。 AWS提供了CloudFront服务&#xff0c;用于将内容缓存在每个用户地理位置本地…

数据光端机设备性能指标介绍

作为安防监控工程&#xff0c;设备的可靠性应该是第一考虑要素。而数据光端机设备的可靠性是设备厂商在产品设计时就必需考虑的&#xff0c;但是&#xff0c;有些厂商可能会因为某些原因而不愿做或不知道怎么做这方面的工作&#xff0c;在这里着重从工程的角度简单地讨论以下问…

光电转换器有什么作用?光纤收发器如何保养?

光电转换器可以使原来的快速以太网平滑升级&#xff0c;并能充分保护用户原来的网络资源&#xff0c;它也可以称为光纤收发器。光电转换器可以实现交换机和计算机之间的互联&#xff0c;也可以作为传输中继&#xff0c;还可以进行单多模转换。光纤收发器在应用过程中&#xff0…

html中输出PHP的下拉列表,html中关于下拉列表select的图文代码详解

HTML中的下拉列表&#xff1a;Html代码VolvoSaabOpelAudi其中select是显示一个下拉列表(drop down list)出来&#xff0c;option是下拉列表中的项目(item)&#xff0c;而option的文本内容(text content)是下拉列表项目中显示到页面上的值&#xff0c;value是真正需要提交到服务…

光纤收发器tx和rx的区别?

光纤收发器&#xff0c;是一种将短距离的双绞线电信号和长距离的光信号进行互换的以太网传输媒体转换单元&#xff0c;在很多地方也被称之为光电转换器&#xff08;Fiber Converter&#xff09;。产品一般应用在以太网电缆无法覆盖、必须使用光纤来延长传输距离的实际网络环境中…

光纤收发器元件级和整机测试内容介绍

对光纤收发器的测试可分为元件级和整机测试&#xff0c;元件级测试主要包括对光纤收发器内部关键器件在电工作的电性能测试。整机测试主要指将光纤收发器接入到以太局域网中&#xff0c;测试整机的功能、性能和特性。那么&#xff0c;具体要怎样测试光纤收发器才是一次完整的测…

如何判断光纤收发器是否有问题?

一般情况下&#xff0c;光纤收发器或光模块的发光功率如下&#xff1a;多模在10db--18db之间&#xff1b;单模20公里在-8db--15db之间&#xff1b;而单模60公里则在-5db--12db之间。但如是光纤收发器的发光功率出现在-30db--45db之间&#xff0c;那么&#xff0c;很有可能这个光…

activemq 实例_在一台计算机上运行多个ActiveMQ实例

activemq 实例几周前&#xff0c;我再次通过Mule ESB解决方案将Apache ActiveMQ用作JMS提供程序。 由于使用ActiveMQ已经有几年了&#xff0c;所以我认为最好检查一些&#xff08;新&#xff09;功能&#xff0c;例如故障转移传输和其他群集功能 。 为了能够测试这些最后的东西…

什么是光纤收发器?光纤收发器作用是什么?

许多朋友一听到光纤收发器这五个大字总会有困惑&#xff0c;比如说什么是光纤收发器&#xff0c;光纤收发器又有什么作用等等疑问。那么&#xff0c;什么是光纤收发器呢&#xff1f;光纤收发器又有什么作用呢&#xff1f;接下来我们就跟随飞畅科技的小编一起来详细了解下吧&…

在15分钟内使用Spring Boot和Spring Security构建一个Web应用程序

“我喜欢编写身份验证和授权代码。” 〜从来没有Java开发人员。 厌倦了一次又一次地建立相同的登录屏幕&#xff1f; 尝试使用Okta API进行托管身份验证&#xff0c;授权和多因素身份验证。 开发人员知道保护Web应用程序安全可能会很麻烦。 正确地做是很难的。 最糟糕的是&…