引言

有没有一种办法可以实现跨应用程序进行通信和跨平台进行通信呢? 换句话说,就是有什么办法可以实现我的应用程序 A 可以和应用程序 B 进行通信呢? 或者说是,我用 Java 写的应用程序和用 . Net 开发的应用程序之间进行通信呢? 很多时候,上面提到的这些,我们是必须要使用的,比如,一个跨应用程序吧。

举一个日常生活的例子吧,我们平常都会使用QQ上面的天气预报功能 吧,那么这个功能是怎么实现的呢?

WebService简介

如果简单的说的话,WebServices就是一组函数库,但是又有不同点,平常我们的函数库都是写在本地上的,而WebService是位于远程机器上(当然也可以是本地机器中)。那么正经一点的说法就是:WebService服务是一种部署在 Web 上的对象或者是应用程序组件,是一种跨编程语言和跨操作系统平台的远程调用技术。

WebService的特点

(1),WebServices 是自包含的。

(2),WebServices 是自我描述的。

(3),WebServices 是跨平台和跨语言的。

(4),WebServices 是基于开放和标准的。

(5),WebServices 是可以组合的。

(6),WebServices 是松散耦合的。

WebService体系结构

Web 服务中介者:

也称为服务代理,用来注册已经发布的 Web服务提供者,并对其进行分类,同时提供搜索服务, 简单来说的话,Web 服务中介者的作用就是把一个 Web 服务请求者和合适的 Web 服务提供者联系在一起,充当一个管理者的角色,一般是通过 UDDI来实现。

Web 服务请求者:

也就是 Web 服务功能的使用者,它通过服务注册中心也 就是 Web 服务中介者查找到所需要的服务,再利用 SOAP 消息向 Web 服务提供者发送请求以获得服务。

Web 服务提供者:

可以发布 Web 服务,并且对使用自身服务的请求 进行响应,Web 服务的拥有者,它会等待其他的 服务或者是应用程序访问自己。

WebServices三种基本元素

SOAP : SOAP 呢,其指导理念是“唯一一个没有发明任何新技术的技术”,是一种用于访问 Web 服务的协议。因为SOAP 基于XML 和 HTTP ,其通过XML 来实现消息描述,然后再通过 HTTP 实现消息传输。

WSDL : WSDL 即Web Services Description Language也就是 Web 服务描述语言。 是基于 XML的用于描述 Web 服务以及如何访问 Web 服务的语言。 WSDL 描述了 Web服务的三个基本属性: (1)服务所提供的操作 (2)如何访问服务 (3)服务位于何处(通过 URL 来确定就 OK 了)

UDDI : UDDI 即 Universal Description,Discovery and Integration,也就是通用的描述,发现以及整合。UDDI 呢是一种目录服务,企业可以通过 UDDI 来注册和搜索 Web 服务。简单来时候话,UDDI 就是一个目录,只不过在这个目录中存放的是一些关于 Web 服务的信息而已。

Apache CXF 简介

Apache CXF = Celtix + XFire,开始叫 Apache CeltiXfire,后来更名为 Apache CXF 了。 CXF 继承了 Celtix 和 XFire 两大开源项目的精华,提供了对 JAX-WS 全面的支持,并且提供 了多种 Binding 、DataBinding、Transport 以及各种 Format 的支持,并且可以根据实际项 目的需要,采用代码优先(Code First)或者 WSDL 优先(WSDL First)来轻松地实现 Web Services 的发布和使用。

Apache CXF特点 : 灵活部署 轻量级容器,可在 Tomcat 或基于 Spring 的容器中部署 Services。 支持多种编程语言 全面支持 JAX-WS 2.0 客户端/服务器编程模型。

CXF使用的相关jar包

我们基于web项目进行的 WebService接口开发,所以 需要导入spring和spring web的相关jar包。 由于cxf-rt-transports-http- jetty.jar依赖jetty的相关jar 包,所以在这里引入。

WebService两大风格

报文格式:

<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<STUDENT>
<NAME>SAM-SHO</NAME>
<StudentPhones>
<StudentPhone>
<num></num>
<type>移动</type>
</StudentPhone>
<StudentPhone>
<num></num>
<type>联通</type>
</StudentPhone>
</StudentPhones>
<StudentAddress>
<homeAddress>苏州高新区</homeAddress>
<workAddress>苏州园区</workAddress>
</StudentAddress>
</STUDENT>

报文格式:

 <soap:Envelope xmlns:soap="http://schemas.xmlsoap.org/soap/envelope/">
<soap:Header>
</soap:Header>
<soap:Body>
<ns2:sayHi xmlns:ns2="http://service.cxf.com/">
<text>hjp</text>
<address>shanghai</address>
</ns2:sayHi>
</soap:Body>
</soap:Envelope>

构建模块:

• 必需的 Envelope 元素,可把此 XML 文档标识为一条 SOAP 消息

• 可选的 Header 元素,包含头部信息

• 必需的 Body 元素,包含所有的调用和响应信息

• 可选的 Fault 元素,提供有关在处理此消息所发生错误的信息

实例:

1).工程右键—New—Interface,添加代码:

说明: @WebService:标记表示该接口是一个WebService服务。

@WebMethod:标记表示WebService中的方法。

@WebParam(name="paramName")表示方法中的参数,name属性限制了参数的名称,若没有指定该属性,参数将会被重命名。

 package com.cxf.service;

 import javax.jws.WebParam;
import javax.jws.WebService; @WebService
public interface HelloWorld { public String sayhi(@WebParam(name="text")String text); }

编写WebService的实现层

 package com.cxf.service;

 import javax.jws.WebService;

 @WebService(endpointInterface="com.cxf.service.HelloWorld" , serviceName="HelloService")
public class HelloWorldImpl implements HelloWorld{ @Override
public String sayhi(String text) { return "say hello,"+text;
} }

说明: @WebService(endpointInterface=”对应的WebService接口”,serviceName=”WebService的名字,自己定义”)。

编写发布WebService的类

 package com.cxf.service;

 import javax.xml.ws.Endpoint;

 public class WebserviceApp {

     public static void main(String[] args) {
System.out.println("start service");
HelloWorldImpl hwi = new HelloWorldImpl();
String address = "http://localhost:8080/helloworld";
Endpoint.publish(address, hwi);
System.out.println("end service"); } }

说明: 使用Endpoint.publish(url,WebService)方法,该方法需要两个参数,一个是服务的发布地址,即最后你在浏览器上访问你发布的WebServcie时用的Url,另一个参数就是你要发布的WebService的实现类。右键——Run as——Java Application,服务发布成功后,访问http://localhost:8080/helloworld?wsdl进行测试,看到如下图效果即为成功。(必须带上?wsdl,不然是不被识别的,它代表的是对这个WebService的相关描述)

查看是否成功发布

编写客户端调用

 public class HelloWorldClient {
 
public static void main(String[] args) {
/*System.out.println("+++++++++start webservice");
ApplicationContext factory = new ClassPathXmlApplicationContext("/applicationContext.xml");
HelloWorld client = (HelloWorld)factory.getBean("client");
String result =client.sayHi("hello spring");
System.out.println(result);*/ //不与spring结合的发布方式
JaxWsProxyFactoryBean factory2 = new JaxWsProxyFactoryBean();
factory2.getInInterceptors().add(new LoggingInInterceptor());
factory2.getOutInterceptors().add(new LoggingOutInterceptor());
factory2.setServiceClass(HelloWorld.class);
factory2.setAddress("http://localhost:8080/helloworld");
HelloWorld client2 = (HelloWorld) factory2.create();
String reply = client2.sayHi("hello world!");
System.out.println(reply);
}
}

用JaxWsProxyFactoryBean方法创建一个客户端厂;factory2.getInInterceptors().add() 和factory2.getOutInterceptors().add()方法,是为了打印控制台日志的。重点是factory2.setServiceClass(HelloWorld.class),表明你的WebService接口是哪一个,然后factory2.setAddress()表明你的服务地址是什么,最后进行客户端client的生成,调用相应的方法

查看调用结果

WebService与Spring的结合

有的WebService服务需要我们根据需求去自主发布,但是有些公用的服务,我们可以在启动Tomact启动时就同时发布上去,在Tomact启动完成后,我们马上就可以进行使用,那么为了实现这个想法,我们就需要用到WebService与Spring的结合。

1).定义服务接口: 同上;

2).编译服务实现层: 同上;

3).在web.xml中增加监听

 <servlet>
<servlet-name>CXFServlet</servlet-name>
<servlet-class>
org.apache.cxf.transport.servlet.CXFServlet
</servlet-class>
<load-on-startup></load-on-startup>
</servlet>
<!-- 对cxf服务进行监听,所有的/webservice/*请求都会被拦截,类似于Spring mvc的拦截器 -->
<servlet-mapping>
<servlet-name>CXFServlet</servlet-name>
<url-pattern>/webservice/*</url-pattern>
</servlet-mapping>

配置文件的引入

配置application文件

新建名为applicationContext.xml的配置文件。

 <?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:jaxws="http://cxf.apache.org/jaxws"
xmlns:jaxrs="http://cxf.apache.org/jaxrs"
xsi:schemaLocation="
http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://cxf.apache.org/jaxws
http://cxf.apache.org/schemas/jaxws.xsd
http://cxf.apache.org/jaxrs
http://cxf.apache.org/schemas/jaxrs.xsd">
<import resource="classpath:META-INF/cxf/cxf.xml"/> <jaxws:endpoint
id="helloWorld"
implementor="com.cxf.service.HelloWorldImpl"
address="/helloWorld" /> <!-- 为了实现spring的客户端调用接口配置 -->
<bean id="client"
class="com.cxf.client.HelloWorldClient"
factory-bean="clientFactory"
factory-method="create"/>
<bean id="clientFactory" class="org.apache.cxf.jaxws.JaxWsProxyFactoryBean">
<property name="serviceClass" value="com.cxf.service.HelloWorld"/>
<property name="address" value="http://localhost:8080/CXFDemo/webservice/helloWorld"/>
</bean>
</beans>

启动Tomact,查看结果

访问的url为localhost:8080(根据实际情况自行修改)/项目名/webservice(web.xml中的配置)/address(application文件中的发布地址)?wsdl。

客户端更改

webservice(CXF)基于3.1.1版本实例的更多相关文章

  1. WebService之基于REST机制的实现实例(Java版)

    REST是REpresentational State Transfer的缩写(一般中文翻译为表述性状态转移).2000年Roy Fielding博士在他的博士论文“Architectural Sty ...

  2. 【转】基于Python的接口测试框架实例

    下面小编就为大家带来一篇基于Python的接口测试框架实例.小编觉得挺不错的,现在就分享给大家,也给大家做个参考.一起跟随小编过来看看吧   背景 最近公司在做消息推送,那么自然就会产生很多接口,测试 ...

  3. 基于Python的接口测试框架实例

    文章来源:http://www.jb51.net/article/96481.htm 下面小编就为大家带来一篇基于Python的接口测试框架实例.小编觉得挺不错的,现在就分享给大家,也给大家做个参考. ...

  4. 基于openstack stable queens版本阅读解析

    基于openstack stable queens版本阅读解析 基于 centos7.5 的linux系统 架构 如下所示,为cinder的官方架构说明: 这里写图片描述 各个组件介绍如下: - DB ...

  5. 在net Core3.1上基于winform实现依赖注入实例

    目录 在net Core3.1上基于winform实现依赖注入实例 1.背景 2.依赖注入 2.1依赖注入是什么? 2.1依赖注入的目的 2.2依赖注入带来的好处 2.2.1生命周期的控制 2.2.2 ...

  6. 基于1.3.3版本tooltip的datagrid单元格tip实现

    基于1.3.3版本tooltip的datagrid单元格tip实现 2013年05月25日 ⁄ datagrid ⁄ 共 6122字 ⁄ 评论数 26 ⁄ 被围观 7,033 views+ 文章目录 ...

  7. centos7发行版号对应基于RHEL Source(版本)对照表

    基础分布 详情地址:https://wiki.centos.org/Download 存档版本 CentOS Linux 7 发布 基于RHEL Source(版本) 存档的树 7(1804) 7.5 ...

  8. 基于jquery判断浏览器版本过低代码

    基于jquery判断浏览器版本过低代码.这是一款对不支持HTML5跟CSS3代码的浏览器提示用户更换特效代码.效果图如下: 在线预览   源码下载 实现的代码. html代码: <div sty ...

  9. 基于zabbix 的memached 多实例监控

    基于zabbix 的memached 多实例监控 zabbix agentd 配置文件新增配置: UserParameter=memcached.server.discovery[*],ps uax ...

随机推荐

  1. 原生js方法document.getElementsByClassName在ie8及其以下的兼容性问题

    document.getElementsByClassName在ie8及其以下浏览器的兼容性问题,在ie8及其以下浏览器中不能使用,针对这个问题,下面给出详细的解决方法,感兴趣的朋友可以参考下     ...

  2. MySQL中TIMESTAMP和DATETIME区别

    1.两者的存储方式不一样 TIMESTAMP:把客户端插入的时间从当前时区转化为UTC(世界标准时间)进行存储.查询时,将其又转化为客户端当前时区进行返回. DATETIME:不做任何改变,基本上是原 ...

  3. 关于timestamp的二三事

    之所以要写timestamp的随笔,是因为之前对它的理解存在误区,so. I have to remind myself by writing this informal essay. 微软文档链接: ...

  4. (转)ThinkPHP自定义标签

    第一:在当前应用下的Conf文件夹中config.php加两个配制项:             'TAGLIB_LOAD' => true,//加载标签库打开             'APP_ ...

  5. MySQL where

    MySQL where 子句 我们知道从MySQL表中使用SQL SELECT 语句来读取数据. 如需有条件地从表中选取数据,可将 WHERE 子句添加到 SELECT 语句中. 语法 以下是SQL ...

  6. 低字节序和高字节序相互转换(Little Endian/Big Endian)

    这个例子展示了如何转换整形数字的字节顺序,该方法可以用来在little-endian和big-endian之间转换. 说明:Windos(x86,x64)和Linux(x86,x64)都是little ...

  7. 【Usaco2008 Mar】土地购买

    [题目描述] 农夫John准备扩大他的农场,他正在考虑N (1 <= N <= 50,000) 块长方形的土地. 每块土地的长宽满足(1 <= 宽 <= 1,000,000; ...

  8. JavaScript-学习一获取表单的值

    <!DOCTYPE html><html><head> <meta charset="utf-8"> <title>he ...

  9. JavaScript设计模式之建造者模式

    一.建造者模式模式概念 建造者模式可以将一个复杂的对象的构建与其表示相分离,使得同样的构建过程可以创建不同的表示.也就是说如果我们用了建造者模式,那么用户就需要指定需要建造的类型就可以得到它们,而具体 ...

  10. 78 Subsets(求子集Medium)

    题目意思:求解一个数组的所有子集,子集内的元素增序排列eg:[1,3,2] result:[],[1],[2],[1,2],[3],[1,3],[2,3],[1,2,3]思路:这是一个递推的过程 [] ...