时间:2022-07-28 11:01:28 | 栏目:JAVA代码 | 点击:次
茫茫人海千千万万,感谢这一秒你看到这里。希望我的文章对你的有所帮助!
愿你在未来的日子,保持热爱,奔赴山海!
webService与我们常见的httpapi接口有什么区别呢?
WebService主要是通过SOAP协议在Web上提供的软件服务,使用WSDL文档进行说明,并通过UDDI进行注册。WebService是一种跨编程语言和跨操作系统平台的远程调用技术,能使得运行在不同机器上的不同应用无须借助附加的、专门的第三方软件或硬件, 就可相互交换数据或集成。依据WebService规范实施的应用之间, 无论它们所使用的语言、 平台或内部协议是什么, 都可以相互交换数据。
所以呢,如果你想是使用不同语言,不同平台,不同地方,想进行数据传输,自我推荐下,选择WebService就没错的!
常伴于三个元素兄弟:UDDI,WSDL,SOAP
话不多说,看太多理论不如自己做个小demo测下来得爽快。所以Let's GO!
做一个天气系统的demo,客户端发送城市名称,服务器端回应相应的天气。
创建一个空项目weatherServer出来,我使用的IDEA版本为2020.3版本。当然一开始我比较详细介绍啦
创建过程基本类似的。如果你想选择使用我,那么我会提议你先创建一个服务端,这样别人可以进行访问,或者使用他人的服务端都OK的啦。这里先会完成创建出一个服务端模块出来,后续的创建客户端模块基本类似。这里我们直接选择一个Maven项目即可,也可以自行选择一个普通Java项目都行。JDK为流行的JDK8即可。
最终得到的空模块服务端如下:
接下来就是开发服务端代码。提供简单的城市天气服务。
定义一个天气业务接口IWeatherService
代码如下:
package com.ws.service; public interface IWeatherSerice { /** * 通过城市名得到对应的天气 * @param city 城市 * @return 天气 */ public String queryWeather(String city); }
编写对应的接口实现类WeatherServiceImpl
对应的代码如下:
package com.ws.service.impl; import com.ws.service.IWeatherSerice; import javax.jws.WebService; @WebService // 用该注解修改表示当前类是一个服务类 必须加上的,不然启动服务的时候会报错。 public class WeatherServiceImpl implements IWeatherSerice { @Override public String queryWeather(String city) { // 这里直接返回对应的城市和晴天!!! return city + "的天气为:晴天!"; } }
创建一个用于发布服务的类WeatherServerDemo。
对应的代码如下:
package com.ws.server; import com.ws.service.impl.WeatherServiceImpl; import javax.xml.ws.Endpoint; public class WeatherServerDemo { public static void main(String[] args) { /** * address: 服务地址 --> 提供访问的地址 * implementor: 服务类 --> 提供访问的接口的实现类 */ Endpoint.publish("http://localhost:8086/weatherServer", new WeatherServiceImpl()); System.out.println("服务发布成功"); } }
运行main方法,开启服务:
可以看到控制台,他没有运行完成后关闭,并且显示服务发布成功了。接下来就在线看看我们发布的服务。
访问刚才提供的访问的地址加上?wsdl,如:http://localhost:8086/weatherServer?wsdl
那如何看这个wsdl文档呢?不怕,让我来教你看几个重要的部分就一目了然啦!注意:wsdl文档需要从下往上看
访问一个详细的参数页面:http://localhost:8086/weatherServer?xsd=1
结构:
为什么需要下载客户端代码,首先前面两个方式需要用到服务端代码的接口,然后如果你是访问远程的服务端,那你也就访问不了。
这里会介绍jdk自带的命令:wsimport
wsimport是jdk自带的webservice客户端工具,可以根据wsdl文档生成客户端调用代码(java代码).
wsimport.exe位于JAVA_HOME\bin目录下
常用参数为:
-d<目录> - 将生成.class文件。默认参数。
-s<目录> - 将生成.java文件。
-p<生成的新包名> -将生成的类,放于指定的包下
wsimport -s ./ http://localhost:8086/weatherServer?wsdl
生成步骤:
在该目录中,进入命令行模式,然后输入wsimport -s ./ http://localhost:8086/weatherServer?wsdl
,下载刚才启动的服务端代码。注意:这一步得保证服务端是启动可访问状态。
最终生成代码的结构在IDEA结构为:
? 这里得到的代码跟wsdl文档对应的名称可以一一对应。当然如果你想改包名的话,可以使用-p参数。
第一种方式:通过得到的代码,service方式调用。
结构如下:
代码如下:
package com.ws.client; import com.ws.service.impl.WeatherServiceImpl; import com.ws.service.impl.WeatherServiceImplService; public class WeatherClientDemo1 { public static void main(String[] args) { // 1. 创建服务视图 WeatherServiceImplService weatherServiceImplService = new WeatherServiceImplService(); // 2. 得到服务实现类 WeatherServiceImpl weatherService = weatherServiceImplService.getPort(WeatherServiceImpl.class); // 3. 调用接口的方法 String result = weatherService.queryWeather("广州"); // 4. 返回远程访问得到的结果 --> result = 广州的天气为:晴天! System.out.println("result = " + result); } }
最终我们运行后得到的结果也正如所预料的一般:
之前这一步有什么缺陷呢,应该很清楚吧:我们固定了一个服务地址了,而且如果要改服务地址的话,可能还需要再生成一遍代码。
所以我们把服务地址写出来,结构如下:
代码如下:
package com.ws.client; import com.ws.service.impl.WeatherServiceImpl; import com.ws.service.impl.WeatherServiceImplService; import javax.xml.namespace.QName; import javax.xml.ws.Service; import java.net.MalformedURLException; import java.net.URL; public class WeatherClientDemo2 { public static void main(String[] args) throws Exception { // 1. 设置访问的服务端地址 URL url = new URL("http://localhost:8086/weatherServer?wsdl"); /* 2.设置服务名称和命名空间 namespaceURI: wsdl的命名空间(targetNamespace) localPart: 是服务视图的名称(service的name值) */ QName qName = new QName("http://impl.service.ws.com/", "WeatherServiceImplService"); // 3. 生成服务视图 Service service = Service.create(url, qName); // 4. 得到服务视图的实现类 --> WeatherServiceImpl WeatherServiceImpl weatherServiceImpl = service.getPort(WeatherServiceImpl.class); // 5. 调用接口方法得到结果! --> result = 深圳的天气为:晴天! String result = weatherServiceImpl.queryWeather("深圳"); System.out.println("result = " + result); } }
这些对应的参数在wsdl的那个位置呢? 看我指给你:
这第二种方式是比较常用的方式,首先推荐!
在不需要下载服务端代码的,通过HTTPURLConnection的方式进行访问服务端。只是这里代码就相对比较长啦,而且需要自行定义XML字符串和解析字符串。
在pom.xml文件中加入一个dom4j的依赖包
<dependencies> <dependency> <groupId>org.dom4j</groupId> <artifactId>dom4j</artifactId> <version>2.1.1</version> </dependency> <dependency> <groupId>jaxen</groupId> <artifactId>jaxen</artifactId> <version>1.1.1</version> </dependency> </dependencies>
结构如下:
创建对应的测试demo,大概结构如下:
代码如下:可以复制到IDEA仔细观看。
package com.ws.client; import org.dom4j.Document; import org.dom4j.DocumentException; import org.dom4j.DocumentHelper; import org.dom4j.Node; import java.io.DataOutputStream; import java.io.InputStream; import java.net.HttpURLConnection; import java.net.URL; import java.util.Scanner; public class WeatherClientDemo3 { public static void main(String[] args) throws Exception { // 1. 设置访问的服务端地址 URL url = new URL("http://localhost:8086/weatherServer?wsdl"); // 2.打开一个通向服务地址的连接 HttpURLConnection connection = (HttpURLConnection) url.openConnection(); // 3.设置参数, --> POST必须大写,否则抛出异常 connection.setRequestMethod("POST"); // 这里是text/xml不是text/html connection.setRequestProperty("content-Type", "text/xml;charset=utf-8"); // 4.设置输入输出,默认是false没有读写的权限 connection.setDoOutput(true); connection.setDoInput(true); // 5.组织SOAP数据,发送请求 String soapXml = getXmlString("佛山"); System.out.println("soapXml = " + soapXml); // 6. 将数据写入到输出流中 DataOutputStream dos = new DataOutputStream(connection.getOutputStream()); dos.write(soapXml.getBytes("utf-8")); dos.flush(); // 7. 判断远程访问是否成功,如果响应码为200即为成功 if (connection.getResponseCode() == 200) { // 8. 获取相应的输入流,得到对应的结果 InputStream ips = connection.getInputStream(); Scanner scanner = new Scanner(ips); StringBuffer buffer = new StringBuffer(); while (scanner.hasNextLine()) { buffer.append(scanner.nextLine()); } scanner.close(); // 得到为xml字符串 System.out.println("buffer = " + buffer); // 9. 解析xml字符串获得返回的字符串 String xml = parseXmlToString(buffer); System.out.println("xml = " + xml); } } private static String parseXmlToString(StringBuffer buffer) { try { // 得到对应的文档对象 Document document = DocumentHelper.parseText(buffer.toString()); // "//"从任意位置的节点上选择名称为 item 的节点。 Node node = document.selectSingleNode("//return"); return node.getText(); } catch (DocumentException e) { e.printStackTrace(); } return null; } private static String getXmlString(String string) { String xml = "<?xml version=\"1.0\" ?>" + "<S:Envelope xmlns:S=\"http://schemas.xmlsoap.org/soap/envelope/\">" + "<S:Body>" + "<ns2:queryWeather xmlns:ns2=\"http://impl.service.ws.com/\">" + "<arg0>" + string + "</arg0>" + "</ns2:queryWeather>" + "</S:Body>" + "</S:Envelope>\""; return xml; } }
得到的结果:
可以看出,可以得到相对应的结果。
因为我是采用XML格式封装数据的嘛,所以在传输过程中,可能需要传输额外的标签,然而标签越来越大的话,导致webservice性能下降。
相信各位看官都对webService如何使用及其它的概念和一些知识点都有了稍稍的了解吧,不确定用到人的是否很多,这里只做入门练习,当然如果我们要整合到SpringBoot项目中,甚至我们可以集成相关依赖来使用webService,例如我们可以使用SpringBoot使用CXF集成WebService。那我们应不应该学习这一门技术呢?其实我觉得看需求,如果需要用到了,我们可以快速了解下,用一用小demo快速入门,接着使用集成CXF来整合到项目中,从而达到这门技术的应用到实际项目中去!
我个人其实也没有接触过,只因项目中需要来调用这样webService接口的需求,所以我也是刚入门,而且现在各网页接口基本都是基于http请求来做的,所以我们可以当了解了解咯,走过路过,瞧一瞧,保证不吃亏呀!!!
让我们也一起加油吧!本人不才,如有什么缺漏、错误的地方,也欢迎各位人才大佬评论中批评指正!当然如果这篇文章确定对你有点小小帮助的话,也请亲切可爱的人才大佬们给个点赞、收藏下吧,一键三连,非常感谢!
学到这里,今天的世界打烊了,晚安!虽然这篇文章完结了,但是我还在,永不完结。我会努力保持写文章。来日方长,何惧车遥马慢!
感谢各位看到这里!愿你韶华不负,青春无悔!