SpringBoot整合WebService的实现示例
作者:NicholasGUB 发布时间:2023-05-25 12:37:55
WebService是一种传统的SOA技术架构,它不依赖于任何的编程语言,也不依赖于任何的技术平台,可以直接基于HTTP协议实现网络应用间的数据交互。
面向服务架构(SOA)是一个组件模型,它将应用程序的不同功能单元(称为服务)进行拆分,并通过这些服务之间定义良好的接口和协议联系起来。接口是采用中立的方式进行定义的,它应该独立于实现服务的硬件平台、操作系统和编程语言。这使得构建在各种各样的系统中的服务可以以一种统一和通用的方式进行交互。
WebService主要用于异构平台之间的整合与调用,例如请求者使用的是Java语言开发,而提供者是Golang语言开发。使用XML进行接口的描述(SOAP协议)。
SpringBoot搭建WebService程序
在springboot-webservice项目中新建3个模块,webservice-server、webservice-client、webservice-common。
webservice-common项目引入项目依赖,webservice-server和webservice-client项目引入webservice-common项目。
<dependency>
<groupId>com.sun.xml.ws</groupId>
<artifactId>jaxws-ri</artifactId>
<version>2.3.5</version>
<type>pom</type>
</dependency>
<dependency>
<groupId>org.hibernate.validator</groupId>
<artifactId>hibernate-validator</artifactId>
<version>7.0.1.Final</version>
</dependency>
<dependency>
<groupId>org.apache.cxf</groupId>
<artifactId>cxf-spring-boot-starter-jaxws</artifactId>
<version>3.4.3</version>
</dependency>
<dependency>
<groupId>org.apache.cxf</groupId>
<artifactId>cxf-rt-transports-http</artifactId>
<version>3.4.3</version>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
<version>2.6.0</version>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web-services</artifactId>
<version>2.6.0</version>
</dependency>
一、定义规范接口
WebService服务端以远程接口为主,在Java实现的WebService技术中,使用CXF开发框架可以直接将接口发布成WebService。
webservice-common模块在com.it.service包下新建MessageService接口。
package com.it.service;
import javax.jws.WebMethod;
import javax.jws.WebParam;
import javax.jws.WebService;
@WebService(name = "MessageService", targetNamespace = "http://service.it.com/")
public interface MessageService {
@WebMethod // webservice方法标注
String echo(@WebParam String message);
}
二、搭建WebService服务端
webservice-server模块定义MessageService的实现子类。
package com.it.service.impl;
import com.it.service.MessageService;
import org.springframework.stereotype.Service;
import javax.jws.WebService;
@WebService(serviceName = "MessageService",
targetNamespace = "http://service.it.com/", // 接口命名空间
endpointInterface = "com.it.service.MessageService") // 接口名称
@Service // 注册到Spring容器
public class MessageServiceImpl implements MessageService {
@Override
public String echo(String message) {
return "[echo]: " + message;
}
}
基于 * 实现安全配置。
package com.it.interceptor;
import lombok.extern.slf4j.Slf4j;
import org.apache.cxf.binding.soap.SoapMessage;
import org.apache.cxf.binding.soap.saaj.SAAJInInterceptor;
import org.apache.cxf.interceptor.Fault;
import org.apache.cxf.phase.AbstractPhaseInterceptor;
import org.apache.cxf.phase.Phase;
import org.springframework.stereotype.Component;
import org.w3c.dom.NodeList;
import javax.xml.soap.SOAPException;
import javax.xml.soap.SOAPHeader;
import javax.xml.soap.SOAPMessage;
import java.util.Objects;
@Slf4j
@Component
public class WebServiceAuthInterceptor extends AbstractPhaseInterceptor<SoapMessage> {
private static final String USER_NAME = "admin"; // 用户名
private static final String USER_PASSWORD = "123456"; //密码
private final SAAJInInterceptor interceptor = new SAAJInInterceptor(); // 创建 *
public WebServiceAuthInterceptor() {
super(Phase.PRE_PROTOCOL);
super.getAfter().add(SAAJInInterceptor.class.getName()); // 添加 *
}
@Override
public void handleMessage(SoapMessage message) throws Fault {
// 获取指定信息
SOAPMessage soapMessage = message.getContent(SOAPMessage.class);
if (soapMessage == null) { // 没有消息
this.interceptor.handleMessage(message); // 走默认流程
soapMessage = message.getContent(SOAPMessage.class); // 再次获取消息
}
SOAPHeader header = null; // SOAP头信息
try {
header = soapMessage.getSOAPHeader();
} catch (SOAPException e) {
e.printStackTrace();
}
if (header == null) { // 没有头信息
throw new Fault(new IllegalAccessException("找不到header信息"));
}
// 解析XML文档(SOAP是XML结构的)
NodeList usernameList = header.getElementsByTagName("username");
NodeList passwordList = header.getElementsByTagName("password");
if (usernameList.getLength() < 1) {
throw new Fault(new IllegalAccessException("找不到header信息"));
}
if (passwordList.getLength() < 1) {
throw new Fault(new IllegalAccessException("找不到header信息"));
}
String username = usernameList.item(0).getTextContent().trim(); // 获取用户名
String password = passwordList.item(0).getTextContent().trim(); // 获取密码
if (Objects.equals(USER_NAME, username) && Objects.equals(USER_PASSWORD, password)) {
log.info("用户访问成功");
} else {
SOAPException soapException = new SOAPException("用户认证失败");
log.error("用户认证失败");
throw new Fault(soapException);
}
}
}
由于当前的webservice是基于CXF开发的,所以需要定义CXF配置类。
package com.it.config;
import com.it.interceptor.WebServiceAuthInterceptor;
import com.it.service.MessageService;
import org.apache.cxf.Bus;
import org.apache.cxf.jaxws.EndpointImpl;
import org.apache.cxf.transport.servlet.CXFServlet;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.web.servlet.ServletRegistrationBean;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import javax.xml.ws.Endpoint;
@Configuration
public class CXFConfig {
private final Bus bus;
private final MessageService messageService;
private final WebServiceAuthInterceptor webServiceAuthInterceptor;
@Autowired
public CXFConfig(Bus bus, MessageService messageService, WebServiceAuthInterceptor webServiceAuthInterceptor) {
this.bus = bus;
this.messageService = messageService;
this.webServiceAuthInterceptor = webServiceAuthInterceptor;
}
public ServletRegistrationBean getServletRegistrationBean() {
return new ServletRegistrationBean(new CXFServlet(), "/services/*"); // 设置webservice访问父路径
}
@Bean
public Endpoint getMessageEndpoint() {
EndpointImpl endpoint = new EndpointImpl(bus, messageService);
endpoint.publish("/MessageService");
endpoint.getInInterceptors().add(webServiceAuthInterceptor); // 添加 *
return endpoint;
}
}
新建SpringBoot启动类,启动程序。
package com.it;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
@SpringBootApplication
public class StartWebServiceServer {
public static void main(String[] args) {
SpringApplication.run(StartWebServiceServer.class, args);
}
}
浏览器访问:http://localhost:8080/services发现webservice发布成功。
WSDL文档也正常出现。
三、搭建WebService客户端
CXF组件下的WebService调用服务使用如下流程:
webservice-client模块创建客户端登录 * ,设置认证信息。
package com.it.interceptor;
import org.apache.cxf.binding.soap.SoapMessage;
import org.apache.cxf.headers.Header;
import org.apache.cxf.helpers.DOMUtils;
import org.apache.cxf.interceptor.Fault;
import org.apache.cxf.phase.AbstractPhaseInterceptor;
import org.apache.cxf.phase.Phase;
import org.w3c.dom.Document;
import org.w3c.dom.Element;
import javax.xml.namespace.QName;
import java.util.List;
public class ClientLoginInterceptor extends AbstractPhaseInterceptor<SoapMessage> {
private final String username;
private final String password;
public ClientLoginInterceptor(String username, String password) {
super(Phase.PRE_PROTOCOL);
this.username = username;
this.password = password;
}
@Override
public void handleMessage(SoapMessage message) throws Fault {
List<Header> headers = message.getHeaders(); // 获取全部头信息
Document document = DOMUtils.createDocument(); // 创建xml文档
Element authElement = document.createElement("authority"); // 认证数据节点
Element usernameElement = document.createElement("username");
Element passwordElement = document.createElement("password");
usernameElement.setTextContent(username);
passwordElement.setTextContent(password);
authElement.appendChild(usernameElement);
authElement.appendChild(passwordElement);
headers.add(0, new Header(new QName("authority"), authElement));
}
}
CXF有两种调用方式,代理调用和动态程序调用。使用代理调用:
package com.it.client;
import com.it.interceptor.ClientLoginInterceptor;
import com.it.service.MessageService;
import org.apache.cxf.jaxws.JaxWsProxyFactoryBean;
public class CXFProxyClient {
private static final String ADDRESS = "http://localhost:8080/services/MessageService?wsdl"; // WebService服务地址
public static void main(String[] args) {
JaxWsProxyFactoryBean jaxWsProxyFactoryBean = new JaxWsProxyFactoryBean();
jaxWsProxyFactoryBean.setAddress(ADDRESS);
jaxWsProxyFactoryBean.setServiceClass(MessageService.class);
jaxWsProxyFactoryBean.getOutInterceptors().add(
new ClientLoginInterceptor("admin","123456") // 设置用户名,密码
);
MessageService messageService = (MessageService)jaxWsProxyFactoryBean.create();
String echo = messageService.echo("[webservice proxy invoke]");
System.out.println("echo = " + echo);
}
}
执行程序,接口调用成功。
动态调用:
package com.it.client;
import com.it.interceptor.ClientLoginInterceptor;
import org.apache.cxf.endpoint.Client;
import org.apache.cxf.jaxws.endpoint.dynamic.JaxWsDynamicClientFactory;
public class CXFDynamicClient {
private static final String ADDRESS = "http://localhost:8080/services/MessageService?wsdl"; // WebService服务地址
public static void main(String[] args) throws Exception {
JaxWsDynamicClientFactory dcf = JaxWsDynamicClientFactory.newInstance();
Client client = dcf.createClient(ADDRESS);
client.getOutInterceptors().add(new ClientLoginInterceptor("admin","123456"));
String message = "dynamic";
Object[] result = client.invoke("echo", message);
System.out.println(result);
}
}
来源:https://blog.csdn.net/Nicholas_GUB/article/details/121587582


猜你喜欢
- 一、简介在上篇 ElasticSearch 文章中,我们详细的介绍了 ElasticSearch 的各种 api 使用。实际的项目开发过程中
- 本文解析了C# KeyUp事件中MessageBox的回车(Enter)键出现回调问题的解决办法。具体问题如下:在一个窗体上有一个名为txt
- 解析XML文件:在Android平台上可以使用SAX、DOM和Android附带的pull解析器解析XML文件;pull解析器提供了各种事件
- 一般来说在Android里要实现树形菜单,都是用ExpandableList(也有高手自己继承ListView或者LinearLayout来
- 我们讲一下Criteria查询,这个对于不是太熟悉SQL语句的我们这些程序员来说是很容易上手的。 废话不多说,看一下例子:&nbs
- 这篇文章主要介绍了Spring Batch批处理框架使用解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需
- SpringMVC用Post方式重定向正常会以return "redirect:/XXX"这种方式直接重定向,但是这种方
- 本文实例讲述了C#实现身份证号码验证的方法。分享给大家供大家参考。具体实现方法如下:随着现在互联网的发展,越来越多的注册用户的地方都用到了身
- 目录一、简述二、内容一、简述利用C# TcpClient在局域网内传输文件,可是文件发送到对面的时候却要重新命名文件的。那可不可以连着文件名
- Java CharArrayReader流一、CharArrayReader流定义API说明:该类实现了一个可用作字符输入流的字符缓冲区,即
- 本文作者:Spring_ZYL文章来源:https://blog.csdn.net/gozhuyinglong版权声明:本文版权归作者所有,
- Toast是一种简易的消息提示框,它无法获取焦点,按设置的时间来显示完以后会自动消失,一般用于帮助或提示。先给大家分享下我的解决思路:不用计
- 觉得作者写得太好了,不得不收藏一下。对这个例子的理解://类型参数不能用基本类型,T和U其实是同一类型。//每次放新数据都成为新的top,把
- 什么是冒泡排序冒泡排序指重复地走访过要排序的元素列,依次比较两个相邻的元素,如果顺序(如从小到大)错误就把他们交换过来。走访元素的工作是重复
- 1.概述在平时的开发中,有一些Jar包因为种种原因,在Maven的中央仓库中没有收录,所以就要使用本地引入的方式加入进来。2. 拷贝至项目根
- 本文实例讲述了Winform下实现图片切换特效的方法,是应用程序开发中非常实用的一个功能。分享给大家供大家参考之用。具体方法如下:本实例源自
- 目录字节输入流字节输入流结构图FileInputStream类构造方法:常用读取方法:字节输出流字节输出流结构图:FileOutputStr
- 方法重写与之前的方法重载不同回顾一下方法重载,相同的方法名不同参数类型和参数数量以及参数顺序package Demo1;import jav
- 前言本文主要给大家介绍了关于C#连接FTP时路径问题的相关内容,分享出来供大家参考学习,话不多说,来一起看看详细的介绍:今天在开发项目时,需
- 一,项目简介经过调查研究进行开发设计的这款仓库管理系统,主要是为商家提供商品货物进销存的信息化管理,以便让商家在竞争如此激烈的今天占据一定的