WebService 简介

首先先说一下,什么是webService,webService也是一种CS结构的WEB服务,C呢就是Client(客户端),S呢就是Server(服务端),webService的服务端必须通过SOAP协议发布出来,然后通过客户端进行调用,webService必须通过XML格式数据报文进行数据传输,比如A服务器发布一个WebService服务,客户端通过A服务发布出来的WSDL描述进行接口调用,WebService是跨语言的,也就是服务端可以用JAVA语言发布,客户端既可以通过PHP调用也可以通过其他语言进行调用,这就是webService的一个大概的简介,下面介绍下webService的一些常用专业名词。

java调用webservice接口 三种方法(WebService的发布与调用)(1)

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服务端发布出来的接口信息,比如有哪些接口方法,入参的详细信息等。

java调用webservice接口 三种方法(WebService的发布与调用)(2)

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();
        }
    }