WebService 简介
首先先说一下,什么是webService,webService也是一种CS结构的WEB服务,C呢就是Client(客户端),S呢就是Server(服务端),webService的服务端必须通过SOAP协议发布出来,然后通过客户端进行调用,webService必须通过XML格式数据报文进行数据传输,比如A服务器发布一个WebService服务,客户端通过A服务发布出来的WSDL描述进行接口调用,WebService是跨语言的,也就是服务端可以用JAVA语言发布,客户端既可以通过PHP调用也可以通过其他语言进行调用,这就是webService的一个大概的简介,下面介绍下webService的一些常用专业名词。
WebService 的关键名词
1.XML:(Extensible Markup Language)扩展型可标记语言。和Json类似,也是一种标准的数据格式,WebService底层必须通过XML格式进行数据传输,因为SOAP协议底层是基于XML格式的,SOAP是XML文档形式的调用方法的规范。
2.Soap:(Simple Object Access Protocol)简单对象存取协议。是XML Web Service 的通信协议。当用户通过UDDI找到你的WSDL描述文档后,他通过SOAP调用你建立的Web服务中的一个或多个操作。SOAP是XML文档形式的调用方法的规范,它可以支持不同的底层接口,像HTTP(S)或者SMTP。
3.WSDL:(Web Services Description Language) WSDL 文件是一个 XML 文档,用于说明WebService服务端发布出来的接口信息,比如有哪些接口方法,入参的详细信息等。
4.UDDI (Universal Description, Discovery, and Integration) 是一个主要针对Web服务供应商和使用者的新项目。在用户能够调用Web服务之前,必须确定这个服务内包含哪些接口方法,找到被调用的接口定义,UDDI是一种根据描述文档来引导系统查找相应服务的机制。UDDI利用SOAP消息机制(标准的XML/HTTP)来发布,编辑,浏览以及查找注册信息。它采用XML格式来封装各种不同类型的数据,并且发送到注册中心或者由注册中心来返回需要的数据。通俗地说就是用来引导客户端去发现服务端发布了哪些接口方法,然后按照WSDL说明通过SOAP协议进行调用的一个过程。
WebService的常用框架
1.JWS是Java语言对WebService服务的一种实现,用来开发和发布服务。而从服务本身的角度来看JWS服务是没有语言界限的。但是Java语言为Java开发者提供便捷发布和调用WebService服务的一种途径,现在没人使用了,已经成为历史
2.XFire是一个高性能的WebService框架,在Java6之前,它的知名度甚至超过了Apache的Axis2,XFire的优点是开发方便,与现有的Web整合很好,可以融为一体,并且开发也很方便。但是对Java之外的语言,没有提供相关的代码工具。XFire后来被Apache收购了,原因是它太优秀了,收购后,随着Java6 JWS的兴起,开源的WebService引擎已经不再被看好,渐渐地都败落了,现在没人使用了,已经成为历史
3.Axis2是Apache下的一个重量级WebService框架,准确说它是一个Web Services / SOAP / WSDL 的引擎,是WebService框架的集大成者,它能不但能制作和发布WebService,而且可以生成Java和其他语言版WebService客户端和服务端代码。这是它的优势所在。但是,这也不可避免地导致了Axis2的复杂性,使用过的开发者都知道,它所依赖的包数量和大小都是很惊人的,打包部署发布都比较麻烦,不能很好的与现有应用整合为一体
4.CXF是Apache旗下一个重磅的SOA简易框架,它实现了ESB(企业服务总线)。CXF来自于XFire项目,经过改造后形成的,就像目前的Struts2来自WebWork一样。可以看出XFire的命运会和WebWork的命运一样,最终会淡出人们的视线。CXF不但是一个优秀的Web Services / SOAP / WSDL 引擎,也是一个不错的ESB总线,为SOA的实施提供了一种选择方案,当然他不是最好的,它仅仅实现了SOA架构的一部分,提供方便的Spring整合方法,可以通过注解、Spring标签式配置来暴露Web Services和消费Web Services
基于SpringBoot集成WebService
上面介绍了WebService是什么以及WebService常用专业词汇还有框架,下面用Java语言为例给大家介绍怎么在springBoot框架中集成WebService的CXF,废话不多说,开始操作吧。
1.maven配置文件添加以下配置信息,如果是Gradle,可以按照Gradle进行配置
<!-- cxf start -->
<dependency>
<groupId>org.apache.cxf</groupId>
<artifactId>cxf-rt-frontend-jaxws</artifactId>
<version>3.2.6</version>
</dependency>
<dependency>
<groupId>org.apache.cxf</groupId>
<artifactId>cxf-rt-transports-http</artifactId>
<version>3.2.6</version>
</dependency>
<!-- cxf end -->
2.CXF注入配置,在springBoot的启动的时候加载该配置,要重点关注下面dispatchServlet方法配置的路径/ele/webService/*和endpoint的/wb,这两个配置是用来连起来访问WSDL说明的,比如这个WebService服务端发布出来的说明就可以这样访问:http://服务ip地址:springBoot启动端口
/ContentPath/ele/webService/wb?wsdl,ContentPath是spring服务的接口前缀,这个不了解可以自行百度,如果没有可以不用配置ContentPath
import org.apache.cxf.Bus;
import org.apache.cxf.bus.spring.SpringBus;
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;
/**
*
* @version v1.0
* @Description: TODO
* @author: guoker
* @date: 2021/7/5 12:08
*/
@Configuration
public class CxfConfig {
@Autowired
private WebServiceTest webServiceTest;
/**
* 此方法作用是改变项目中服务名的前缀名,此处127.0.0.1或者localhost不能访问时,请使用ipconfig查看本机ip来访问
* 此方法被注释后:wsdl访问地址为http://127.0.0.1:8080/services/user?wsdl
* 去掉注释后:wsdl访问地址为:http://127.0.0.1:8080/soap/user?wsdl
*
* @return
*/
@Bean
public ServletRegistrationBean dispatchServlet() {
ServletRegistrationBean servletRegistrationBean = new ServletRegistrationBean(new CXFServlet(), "/ele/webService/*");
return servletRegistrationBean;
}
@Bean(name = Bus.DEFAULT_BUS_ID)
public SpringBus springBus()
{
return new SpringBus();
}
/** JAX-WS
* 站点服务
* **/
@Bean
public Endpoint endpoint() {
EndpointImpl endpoint = new EndpointImpl(springBus(), webServiceTest);
endpoint.publish("/wb");
return endpoint;
}
3.编写发布的接口服务
import javax.jws.WebMethod;
import javax.jws.WebParam;
import javax.jws.WebResult;
import javax.jws.WebService;
@WebService(name = "WebServiceTest", targetNamespace = "http://service.webservice.kexin.com")
public interface WebServiceTest {
@WebMethod
@WebResult(name = "String", targetNamespace = "")
String getUserByName(@WebParam(name = "username") String username, @WebParam(name = "userpwd") String userpwd, @WebParam(name = "usertype") String usertype) throws Exception;
}
4.编写发布接口的实现类
import org.springframework.stereotype.Component;
import javax.jws.WebService;
/**
*
* @version v1.0
* @Description: TODO
* @author: guoker
* @date: 2021/7/5 11:31
*/
@WebService(endpointInterface = "com.dxhy.bxsdk.business.service.WebServiceTest",
targetNamespace = "http://service.webservice.kexin.com", serviceName = "WebServiceTest")
@Component
public class WebServiceTestImp implements WebServiceTest {
@Override
public String getUserByName(String username, String userpwd, String usertype) throws Exception {
return "hello WebService";
}
}
WebService客户端调用示例
/**
* 2:动态调用
*/
public static void main() {
// 创建动态客户端
JaxWsDynamicClientFactory dcf = JaxWsDynamicClientFactory.newInstance();
Client client = dcf.createClient("http://127.0.0.1:8184/bx/ele/webService/wb?wsdl");
// 需要密码的情况需要加上用户名和密码
// client.getOutInterceptors().add(new ClientLoginInterceptor(USER_NAME, PASS_WORD));
Object[] objects = new Object[0];
try {
// invoke("方法名",参数1,参数2,参数3....);
objects = client.invoke("getUserByName", "guoker", "123456", "测试类型");
System.out.println("返回数据:" + objects[0]);
} catch (java.lang.Exception e) {
e.printStackTrace();
}
}