- ubuntu12.04环境下使用kvm ioctl接口实现最简单的虚拟机
- Ubuntu 通过无线网络安装Ubuntu Server启动系统后连接无线网络的方法
- 在Ubuntu上搭建网桥的方法
- ubuntu 虚拟机上网方式及相关配置详解
CFSDN坚持开源创造价值,我们致力于搭建一个资源共享平台,让每一个IT人在这里找到属于你的精彩世界.
这篇CFSDN的博客文章spring boot 开发soap webservice的实现代码由作者收集整理,如果你对这篇文章有兴趣,记得点赞哟.
介绍 。
spring boot web模块提供了restcontroller实现restful,第一次看到这个名字的时候以为还有soapcontroller,很可惜没有,对于soap webservice提供了另外一个模块spring-boot-starter-web-services支持。本文介绍如何在spring boot中开发soap webservice接口,以及接口如何同时支持soap和restful两种协议.
soap webservice 。
web service是一个平台独立的,低耦合的,自包含的、基于可编程的web的应用程序,既可以是soap webservice也可以是rest webservice,在rest还没出来之前,我们说webservice一般是指基于soap协议进行通信的web应用程序.
在开始之前,我觉得有必要了解下soap webservice,具体的概念网上可以找到很多资料,但网上资料概念性较强,而且soap协议使用的是xml进行通信,相信xml里面一个namespace就能吓跑一大堆人,所以这里不讨论具体的soap协议细节,我想通过一个例子来说明什么是soap webservice,通过该例子,你能了解soap webservice其运作原理,当然如果你觉得你对这个已经很了解了,大可跳过本章节,本章节跟后面的内容没有任何关系.
假设我们开发了一个web接口,想给别人用,我们要怎么办 。
那问题来了,我们能不能只把接口部署到服务器上,然后接口不单能提供具体的服务,而且还能自动生成一份标准的接口文档,把接口信息都记录在该文档里,如果能做到,是不是能做到"接口即文档"的目的.
那么一个接口的信息包括哪些呢?
soap webservice里wsdl文件就是接口描述信息。核心的信息就是以上几个.
第二个问题,由于web service是一个平台独立,也就是说,使用接口的人不知道这个service是用什么技术开发的,可能是php可能是java等,但接口的参数和返回的数据都是一样的,要达到这种目的,就需要两个东西,一个是跟平台无关的数据格式,soap使用的是xml,一个是通信协议,也就是soap协议.
下面就介绍如何不使用任何框架,仅通过servlet实现一个webservice。该webservice功能很简单,就是通过一个人的姓名查询这个人的详细信息.
ps:servlet是java web的基础,理解servlet对理解整个java web非常重要,没写过servlet就开始用各种框架写接口就是在胡闹.
1. wsdl文件 。
准备以下wsdl文件,不要管这个文件是怎么来的,是怎么生成的,我们这次只讲原理,不谈细节,总之,你根据需求写出了这个wsdl文件.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
|
<?xml version=
"1.0"
encoding=
"utf-8"
standalone=
"no"
?><wsdl:definitions xmlns:wsdl=
"http://schemas.xmlsoap.org/wsdl/"
xmlns:sch=
"http://www.definesys.com/xml/employee"
xmlns:soap=
"http://schemas.xmlsoap.org/wsdl/soap/"
xmlns:tns=
"http://www.definesys.com/xml/employee"
targetnamespace=
"http://www.definesys.com/xml/employee"
>
<wsdl:types>
<xs:schema xmlns:xs=
"http://www.w3.org/2001/xmlschema"
elementformdefault=
"qualified"
targetnamespace=
"http://www.definesys.com/xml/employee"
>
<xs:element name=
"employeedetailrequest"
>
<xs:complextype>
<xs:sequence>
<xs:element name=
"name"
type=
"xs:string"
/>
</xs:sequence>
</xs:complextype>
</xs:element>
<xs:element name=
"employeedetailresponse"
>
<xs:complextype>
<xs:sequence>
<xs:element name=
"employee"
type=
"tns:employee"
/>
</xs:sequence>
</xs:complextype>
</xs:element>
<xs:complextype name=
"employee"
>
<xs:sequence>
<xs:element name=
"name"
type=
"xs:string"
/>
<xs:element name=
"email"
type=
"xs:string"
/>
</xs:sequence>
</xs:complextype>
</xs:schema>
</wsdl:types>
<wsdl:message name=
"employeedetailrequest"
>
<wsdl:part element=
"tns:employeedetailrequest"
name=
"employeedetailrequest"
>
</wsdl:part>
</wsdl:message>
<wsdl:message name=
"employeedetailresponse"
>
<wsdl:part element=
"tns:employeedetailresponse"
name=
"employeedetailresponse"
>
</wsdl:part>
</wsdl:message>
<wsdl:porttype name=
"employee"
>
<wsdl:operation name=
"employeedetail"
>
<wsdl:input message=
"tns:employeedetailrequest"
name=
"employeedetailrequest"
>
</wsdl:input>
<wsdl:output message=
"tns:employeedetailresponse"
name=
"employeedetailresponse"
>
</wsdl:output>
</wsdl:operation>
</wsdl:porttype>
<wsdl:binding name=
"employeesoap11"
type=
"tns:employee"
>
<soap:binding style=
"document"
transport=
"http://schemas.xmlsoap.org/soap/http"
/>
<wsdl:operation name=
"employeedetail"
>
<soap:operation soapaction=
""
/>
<wsdl:input name=
"employeedetailrequest"
>
<soap:body use=
"literal"
/>
</wsdl:input>
<wsdl:output name=
"employeedetailresponse"
>
<soap:body use=
"literal"
/>
</wsdl:output>
</wsdl:operation>
</wsdl:binding>
<wsdl:service name=
"employeeservice"
>
<wsdl:port binding=
"tns:employeesoap11"
name=
"employeesoap11"
>
<soap:address location=
"http://localhost:8081/ws-servlet/ws/employee-detail"
/>
</wsdl:port>
</wsdl:service>
</wsdl:definitions>
|
soap:address location里面端口号需要修改为servlet运行的端口号.
从以下xml片段可以看出 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
|
...
<wsdl:binding name=
"employeesoap11"
type=
"tns:employee"
>
<soap:binding style=
"document"
transport=
"http://schemas.xmlsoap.org/soap/http"
/>
<wsdl:operation name=
"employeedetail"
>
<soap:operation soapaction=
""
/>
<wsdl:input name=
"employeedetailrequest"
>
<soap:body use=
"literal"
/>
</wsdl:input>
<wsdl:output name=
"employeedetailresponse"
>
<soap:body use=
"literal"
/>
</wsdl:output>
</wsdl:operation>
</wsdl:binding>
<wsdl:service name=
"employeeservice"
>
<wsdl:port binding=
"tns:employeesoap11"
name=
"employeesoap11"
>
<soap:address location=
"http://localhost:8081/ws-servlet/ws/employee-detail"
/>
</wsdl:port>
</wsdl:service>
|
2. 获取wsdl文件servlet 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
|
package
com.definesys.demo.servlet;
import
javax.servlet.servletexception;
import
javax.servlet.http.httpservlet;
import
javax.servlet.http.httpservletrequest;
import
javax.servlet.http.httpservletresponse;
import
java.io.ioexception;
/**
* @copyright: shanghai definesys company.all rights reserved.
* @description:
* @author: jianfeng.zheng
* @since: 2019/1/5 下午1:45
* @history: 1.2019/1/5 created by jianfeng.zheng
*/
public
class
wsdlservlet
extends
httpservlet {
public
static
final
string wsdl_xml =
"<?xml version=\"1.0\" encoding=\"utf-8\" standalone=\"no\"?><wsdl:definitions xmlns:wsdl=\"http://schemas.xmlsoap.org/wsdl/\" xmlns:sch=\"http://www.definesys.com/xml/employee\" xmlns:soap=\"http://schemas.xmlsoap.org/wsdl/soap/\" xmlns:tns=\"http://www.definesys.com/xml/employee\" targetnamespace=\"http://www.definesys.com/xml/employee\">\n"
+
" <wsdl:types>\n"
+
" <xs:schema xmlns:xs=\"http://www.w3.org/2001/xmlschema\" elementformdefault=\"qualified\" targetnamespace=\"http://www.definesys.com/xml/employee\">\n"
+
"\n"
+
" <xs:element name=\"employeedetailrequest\">\n"
+
" <xs:complextype>\n"
+
" <xs:sequence>\n"
+
" <xs:element name=\"name\" type=\"xs:string\"/>\n"
+
" </xs:sequence>\n"
+
" </xs:complextype>\n"
+
" </xs:element>\n"
+
"\n"
+
" <xs:element name=\"employeedetailresponse\">\n"
+
" <xs:complextype>\n"
+
" <xs:sequence>\n"
+
" <xs:element name=\"employee\" type=\"tns:employee\"/>\n"
+
" </xs:sequence>\n"
+
" </xs:complextype>\n"
+
" </xs:element>\n"
+
"\n"
+
" <xs:complextype name=\"employee\">\n"
+
" <xs:sequence>\n"
+
" <xs:element name=\"name\" type=\"xs:string\"/>\n"
+
" <xs:element name=\"email\" type=\"xs:string\"/>\n"
+
" </xs:sequence>\n"
+
" </xs:complextype>\n"
+
"\n"
+
"</xs:schema>\n"
+
" </wsdl:types>\n"
+
" <wsdl:message name=\"employeedetailrequest\">\n"
+
" <wsdl:part element=\"tns:employeedetailrequest\" name=\"employeedetailrequest\">\n"
+
" </wsdl:part>\n"
+
" </wsdl:message>\n"
+
" <wsdl:message name=\"employeedetailresponse\">\n"
+
" <wsdl:part element=\"tns:employeedetailresponse\" name=\"employeedetailresponse\">\n"
+
" </wsdl:part>\n"
+
" </wsdl:message>\n"
+
" <wsdl:porttype name=\"employee\">\n"
+
" <wsdl:operation name=\"employeedetail\">\n"
+
" <wsdl:input message=\"tns:employeedetailrequest\" name=\"employeedetailrequest\">\n"
+
" </wsdl:input>\n"
+
" <wsdl:output message=\"tns:employeedetailresponse\" name=\"employeedetailresponse\">\n"
+
" </wsdl:output>\n"
+
" </wsdl:operation>\n"
+
" </wsdl:porttype>\n"
+
" <wsdl:binding name=\"employeesoap11\" type=\"tns:employee\">\n"
+
" <soap:binding style=\"document\" transport=\"http://schemas.xmlsoap.org/soap/http\"/>\n"
+
" <wsdl:operation name=\"employeedetail\">\n"
+
" <soap:operation soapaction=\"\"/>\n"
+
" <wsdl:input name=\"employeedetailrequest\">\n"
+
" <soap:body use=\"literal\"/>\n"
+
" </wsdl:input>\n"
+
" <wsdl:output name=\"employeedetailresponse\">\n"
+
" <soap:body use=\"literal\"/>\n"
+
" </wsdl:output>\n"
+
" </wsdl:operation>\n"
+
" </wsdl:binding>\n"
+
" <wsdl:service name=\"employeeservice\">\n"
+
" <wsdl:port binding=\"tns:employeesoap11\" name=\"employeesoap11\">\n"
+
" <soap:address location=\"http://localhost:8081/ws-servlet/ws/employee-detail\"/>\n"
+
" </wsdl:port>\n"
+
" </wsdl:service>\n"
+
"</wsdl:definitions>"
;
@override
protected
void
doget(httpservletrequest req, httpservletresponse resp)
throws
servletexception, ioexception {
resp.setcontenttype(
"text/xml"
);
resp.getoutputstream().write(wsdl_xml.getbytes());
}
}
|
是不是很简单,是的,为了简单,我直接将wsdl文件用变量存储,我们还需要配置下web.xml 。
web.xml 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
<?xml version=
"1.0"
encoding=
"utf-8"
?>
<web-app xmlns=
"http://xmlns.jcp.org/xml/ns/javaee"
xmlns:xsi=
"http://www.w3.org/2001/xmlschema-instance"
xsi:schemalocation=
"http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_3_1.xsd"
version=
"3.1"
>
<servlet>
<servlet-name>wsdl</servlet-name>
<servlet-
class
>com.definesys.demo.servlet.wsdlservlet</servlet-
class
>
</servlet>
<servlet-mapping>
<servlet-name>wsdl</servlet-name>
<url-pattern>/ws/employee</url-pattern>
</servlet-mapping>
</web-app>
|
这样我们访问http://localhost:8080/ws/employee就能返回一个wsdl文件,也就是接口描述文件。在wsdl文件里,我们定义接口地址为http://localhost:8080/ws/employee-detail,接下来我们就要实现这个接口.
3. 业务servlet 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
|
import
javax.servlet.servletexception;
import
javax.servlet.http.httpservlet;
import
javax.servlet.http.httpservletrequest;
import
javax.servlet.http.httpservletresponse;
import
java.io.ioexception;
/**
* @copyright: shanghai definesys company.all rights reserved.
* @description:
* @author: jianfeng.zheng
* @since: 2019/1/5 下午2:56
* @history: 1.2019/1/5 created by jianfeng.zheng
*/
public
class
employeeservlet
extends
httpservlet {
@override
protected
void
dopost(httpservletrequest req, httpservletresponse resp)
throws
servletexception, ioexception {
string response =
"<soap-env:envelope xmlns:soap-env=\"http://schemas.xmlsoap.org/soap/envelope/\">\n"
+
" <soap-env:header/>\n"
+
" <soap-env:body>\n"
+
" <ns2:employeedetailresponse xmlns:ns2=\"http://www.definesys.com/xml/employee\">\n"
+
" <ns2:employee>\n"
+
" <ns2:name>jianfeng</ns2:name>\n"
+
" <ns2:email>jianfeng.zheng@definesys.com</ns2:email>\n"
+
" </ns2:employee>\n"
+
" </ns2:employeedetailresponse>\n"
+
" </soap-env:body>\n"
+
"</soap-env:envelope>"
;
resp.getoutputstream().write(response.getbytes());
}
}
|
这里不做任何业务处理,不做xml转bean,不做bean转xml,就是这么暴力,直接返回xml,但他仍是一个soap服务,支持所有soap工具调用.
将servlet配置到web.xml里 。
web.xml 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
|
<?xml version=
"1.0"
encoding=
"utf-8"
?>
<web-app xmlns=
"http://xmlns.jcp.org/xml/ns/javaee"
xmlns:xsi=
"http://www.w3.org/2001/xmlschema-instance"
xsi:schemalocation=
"http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_3_1.xsd"
version=
"3.1"
>
<servlet>
<servlet-name>wsdl</servlet-name>
<servlet-
class
>com.definesys.demo.servlet.wsdlservlet</servlet-
class
>
</servlet>
<servlet>
<servlet-name>employee</servlet-name>
<servlet-
class
>com.definesys.demo.servlet.employeeservlet</servlet-
class
>
</servlet>
<servlet-mapping>
<servlet-name>wsdl</servlet-name>
<url-pattern>/ws/employee</url-pattern>
</servlet-mapping>
<servlet-mapping>
<servlet-name>employee</servlet-name>
<url-pattern>/ws/employee-detail</url-pattern>
</servlet-mapping>
</web-app>
|
/ws/employee-detail这个地址必须和wsdl文件里定义的保持一致,不然服务无法被找到.
4. 测试 。
使用soapui测试我们的webservice,通过地址http://localhost:8081/ws-servlet/ws/employee导入wsdl文件,测试接口,返回我们在业务servlet里面写死的内容。恭喜你,你已经不依赖任何第三方包完成了一个soap webservice.
当然这个只是一个玩具,但框架就是在上面的基础上进行扩展,增加wsdl文件自动生成,xml转java,java转xml,xml校验,错误处理等功能,如果你有时间,你也可以写一个soap webservice框架.
代码已经上传至github,欢迎star,开始进入正题,偏的有点远.
spring boot开发soap webservice 。
1. 创建spring boot工程 。
你可以通过spring initializr初始化spring boot工程,也可以通过inte idea的spring initializr插件进行初始化,个人推荐后面这种.
2. 添加依赖 。
添加soap webservice相关依赖包和插件, 。
pom.xml 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
|
<!--依赖-->
<dependency>
<groupid>org.springframework.boot</groupid>
<artifactid>spring-boot-starter-web-services</artifactid>
</dependency>
<dependency>
<groupid>wsdl4j</groupid>
<artifactid>wsdl4j</artifactid>
</dependency>
...
<!--插件-->
<plugin>
<groupid>org.codehaus.mojo</groupid>
<artifactid>jaxb2-maven-plugin</artifactid>
<version>
1.6
</version>
<executions>
<execution>
<id>xjc</id>
<goals>
<goal>xjc</goal>
</goals>
</execution>
</executions>
<configuration>
<schemadirectory>${project.basedir}/src/main/resources/</schemadirectory>
<!--<schemafiles>employee.xsd</schemafiles>-->
<outputdirectory>${project.basedir}/src/main/java</outputdirectory>
<packagename>com.definesys.tutorial.ws.type</packagename>
<clearoutputdir>
false
</clearoutputdir>
</configuration>
</plugin>
|
插件jaxb2能够实现java和xml之间互转,下面是几个参数的说明 。
3. 编写xsd文件 。
假设我们的需求是通过员工工号查询员工详细信息,根据需求编写以下xsd文件,并保存在/src/main/resources/目录下.
employee.xsd 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
|
<xs:schema xmlns:xs=
"http://www.w3.org/2001/xmlschema"
xmlns:tns=
"http://www.definesys.com/xml/employee"
targetnamespace=
"http://www.definesys.com/xml/employee"
elementformdefault=
"qualified"
>
<xs:element name=
"employeedetailrequest"
>
<xs:complextype>
<xs:sequence>
<xs:element name=
"code"
type=
"xs:string"
/>
</xs:sequence>
</xs:complextype>
</xs:element>
<xs:element name=
"employeedetailresponse"
>
<xs:complextype>
<xs:sequence>
<xs:element name=
"employee"
type=
"tns:employee"
/>
</xs:sequence>
</xs:complextype>
</xs:element>
<xs:complextype name=
"employee"
>
<xs:sequence>
<xs:element name=
"code"
type=
"xs:string"
/>
<xs:element name=
"name"
type=
"xs:string"
/>
<xs:element name=
"email"
type=
"xs:string"
/>
</xs:sequence>
</xs:complextype>
</xs:schema>
|
4. 生成java类型文件 。
我们需要根据xsd文件生成java类型文件,这就要借助maven插件jaxb2,打开终端运行命令mvn jaxb2:xjc,如果运行正常,就会在目录com.definesys.tutorial.ws.type下生成一堆java文件,此时文件结构如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
|
.
├── java
│ └── com
│ └── definesys
│ └── tutorial
│ └── ws
│ ├── springbootwsapplication.java
│ └── type
│ ├── employee.java
│ ├── employeedetailrequest.java
│ ├── employeedetailresponse.java
│ ├── objectfactory.java
│ └──
package
-info.java
└── resources
├── application.properties
├── employee.xsd
├──
static
└── templates
|
5. 创建配置文件 。
webserviceconfig.java 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
|
package
com.definesys.tutorial.ws;
import
org.springframework.boot.web.servlet.servletregistrationbean;
import
org.springframework.context.applicationcontext;
import
org.springframework.context.annotation.bean;
import
org.springframework.context.annotation.configuration;
import
org.springframework.core.io.classpathresource;
import
org.springframework.ws.config.annotation.enablews;
import
org.springframework.ws.config.annotation.wsconfigureradapter;
import
org.springframework.ws.transport.http.messagedispatcherservlet;
import
org.springframework.ws.wsdl.wsdl11.defaultwsdl11definition;
import
org.springframework.ws.wsdl.wsdl11.wsdl11definition;
import
org.springframework.xml.xsd.simplexsdschema;
import
org.springframework.xml.xsd.xsdschema;
/**
* @copyright: shanghai definesys company.all rights reserved.
* @description:
* @author: jianfeng.zheng
* @since: 2019/1/5 下午4:46
* @history: 1.2019/1/5 created by jianfeng.zheng
*/
@enablews
@configuration
public
class
webserviceconfig
extends
wsconfigureradapter {
@bean
public
servletregistrationbean messagedispatcherservlet(applicationcontext applicationcontext) {
messagedispatcherservlet servlet =
new
messagedispatcherservlet();
servlet.setapplicationcontext(applicationcontext);
servlet.settransformwsdllocations(
true
);
return
new
servletregistrationbean(servlet,
"/ws/*"
);
}
@bean
(name =
"employee"
)
public
wsdl11definition defaultwsdl11definition(xsdschema schema) {
defaultwsdl11definition wsdl =
new
defaultwsdl11definition();
wsdl.setporttypename(
"employeeport"
);
wsdl.setlocationuri(
"/ws/employee-detail"
);
wsdl.settargetnamespace(
"http://www.definesys.com/xml/employee"
);
wsdl.setschema(schema);
return
wsdl;
}
@bean
public
xsdschema employeeschema() {
return
new
simplexsdschema(
new
classpathresource(
"employee.xsd"
));
}
}
|
6. 创建业务服务 。
employeesoapcontroller.java 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
|
package
com.definesys.tutorial.ws;
import
com.definesys.tutorial.ws.type.employee;
import
com.definesys.tutorial.ws.type.employeedetailrequest;
import
com.definesys.tutorial.ws.type.employeedetailresponse;
import
org.springframework.ws.server.endpoint.annotation.payloadroot;
import
org.springframework.ws.server.endpoint.annotation.requestpayload;
import
org.springframework.ws.server.endpoint.annotation.responsepayload;
/**
* @copyright: shanghai definesys company.all rights reserved.
* @description:
* @author: jianfeng.zheng
* @since: 2019/1/5 下午4:49
* @history: 1.2019/1/5 created by jianfeng.zheng
*/
@endpoint
public
class
employeesoapcontroller {
private
static
final
string namespace_uri =
"http://www.definesys.com/xml/employee"
;
@payloadroot
(namespace = namespace_uri, localpart =
"employeedetailrequest"
)
@responsepayload
public
employeedetailresponse getemployee(
@requestpayload
employeedetailrequest request) {
employeedetailresponse response =
new
employeedetailresponse();
//这里只作为演示,真正开发中需要编写业务逻辑代码
employee employee =
new
employee();
employee.setname(
"jianfeng"
);
employee.setemail(
"jianfeng.zheng@definesys.com"
);
employee.setcode(request.getcode());
response.setemployee(employee);
return
response;
}
}
|
与restcontroller不一样的是,spring boot soap是根据请求报文来指定调用的函数,restcontroller是根据请求路径来确定。@payloadroot就是关键,如本次请求报文如下:
1
2
3
4
5
6
7
8
|
<soapenv:envelope xmlns:soapenv=
"http://schemas.xmlsoap.org/soap/envelope/"
xmlns:emp=
"http://www.definesys.com/xml/employee"
>
<soapenv:header/>
<soapenv:body>
<emp:employeedetailrequest>
<emp:code>?</emp:code>
</emp:employeedetailrequest>
</soapenv:body>
</soapenv:envelope>
|
xmlns:emp="http://www.definesys.com/xml/employee"就是@payloadroot.namespace,emp:employeedetailrequest对应@payloadroot.localpart。理解了这个其他都很好理解.
7. 测试 。
使用soapui进行测试,通过地址http://localhost:8080/ws/employee.wsdl导入wsdl文件进行测试.
输入报文 。
1
2
3
4
5
6
7
8
|
<soapenv:envelope xmlns:soapenv=
"http://schemas.xmlsoap.org/soap/envelope/"
xmlns:emp=
"http://www.definesys.com/xml/employee"
>
<soapenv:header/>
<soapenv:body>
<emp:employeedetailrequest>
<emp:code>
004
</emp:code>
</emp:employeedetailrequest>
</soapenv:body>
</soapenv:envelope>
|
输出报文 。
1
2
3
4
5
6
7
8
9
10
11
12
|
<soap-env:envelope xmlns:soap-env=
"http://schemas.xmlsoap.org/soap/envelope/"
>
<soap-env:header/>
<soap-env:body>
<ns2:employeedetailresponse xmlns:ns2=
"http://www.definesys.com/xml/employee"
>
<ns2:employee>
<ns2:code>
004
</ns2:code>
<ns2:name>jianfeng</ns2:name>
<ns2:email>jianfeng.zheng
@definesys
.com</ns2:email>
</ns2:employee>
</ns2:employeedetailresponse>
</soap-env:body>
</soap-env:envelope>
|
同时提供soap和restful两种服务 。
soap一般在企业内部用的比较多,做系统间的集成,restful一般用于移动应用和h5应用,如果在企业应用开发里能够同时提供两种协议的支持,将极大提高接口的复用。其实也没有想象中的那么复杂,在本例中,只需把业务逻辑部分用service实现再创建一个restcontroller即可,通过设计模式即可解决,不需要引入新的技术.
employeeservice.java 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
|
package
com.definesys.tutorial.ws;
import
com.definesys.tutorial.ws.type.employee;
import
com.definesys.tutorial.ws.type.employeedetailrequest;
import
com.definesys.tutorial.ws.type.employeedetailresponse;
import
org.springframework.stereotype.service;
/**
* @copyright: shanghai definesys company.all rights reserved.
* @description:
* @author: jianfeng.zheng
* @since: 2019/1/5 下午5:42
* @history: 1.2019/1/5 created by jianfeng.zheng
*/
@service
public
class
employeeservice {
public
employeedetailresponse getemployee(employeedetailrequest request) {
employeedetailresponse response =
new
employeedetailresponse();
//这里只作为演示,真正开发中需要编写业务逻辑代码
employee employee =
new
employee();
employee.setname(
"jianfeng"
);
employee.setemail(
"jianfeng.zheng@definesys.com"
);
employee.setcode(request.getcode());
response.setemployee(employee);
return
response;
}
}
|
employeesoapcontroller.java 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
|
package
com.definesys.tutorial.ws;
import
com.definesys.tutorial.ws.type.employee;
import
com.definesys.tutorial.ws.type.employeedetailrequest;
import
com.definesys.tutorial.ws.type.employeedetailresponse;
import
org.springframework.beans.factory.annotation.autowired;
import
org.springframework.ws.server.endpoint.annotation.endpoint;
import
org.springframework.ws.server.endpoint.annotation.payloadroot;
import
org.springframework.ws.server.endpoint.annotation.requestpayload;
import
org.springframework.ws.server.endpoint.annotation.responsepayload;
/**
* @copyright: shanghai definesys company.all rights reserved.
* @description:
* @author: jianfeng.zheng
* @since: 2019/1/5 下午4:49
* @history: 1.2019/1/5 created by jianfeng.zheng
*/
@endpoint
public
class
employeesoapcontroller {
@autowired
private
employeeservice service;
private
static
final
string namespace_uri =
"http://www.definesys.com/xml/employee"
;
@payloadroot
(namespace = namespace_uri, localpart =
"employeedetailrequest"
)
@responsepayload
public
employeedetailresponse getemployee(
@requestpayload
employeedetailrequest request) {
return
service.getemployee(request);
}
}
|
employeerestcontroller.java 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
|
package
com.definesys.tutorial.ws;
import
com.definesys.tutorial.ws.type.employeedetailrequest;
import
com.definesys.tutorial.ws.type.employeedetailresponse;
import
org.springframework.beans.factory.annotation.autowired;
import
org.springframework.web.bind.annotation.requestbody;
import
org.springframework.web.bind.annotation.requestmapping;
import
org.springframework.web.bind.annotation.requestmethod;
import
org.springframework.web.bind.annotation.restcontroller;
/**
* @copyright: shanghai definesys company.all rights reserved.
* @description:
* @author: jianfeng.zheng
* @since: 2019/1/5 下午5:43
* @history: 1.2019/1/5 created by jianfeng.zheng
*/
@restcontroller
@requestmapping
(value =
"/rest"
)
public
class
employeerestcontroller {
@autowired
private
employeeservice service;
@requestmapping
(value =
"/employee-detail"
, method = requestmethod.post)
public
employeedetailresponse getemployeedetail(
@requestbody
employeedetailrequest request) {
return
service.getemployee(request);
}
}
|
测试 。
1
2
3
4
5
6
7
8
|
$ curl http:
//localhost:8080/rest/employee-detail -x post -d '{"code":"004"}' -h "content-type: application/json"
{
"employee"
: {
"code"
:
"004"
,
"name"
:
"jianfeng"
,
"email"
:
"jianfeng.zheng@definesys.com"
}
}
|
这样就实现了soap和rest同时提供的目的.
本文代码已提交至gitlab欢迎star 。
相关参考文档 。
https://spring.io/guides/gs/producing-web-service/ https://github.com/wls1036/tutorial-springboot-soap https://github.com/wls1036/pure-ws-servlet 。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我.
原文链接:https://segmentfault.com/a/1190000017777211 。
最后此篇关于spring boot 开发soap webservice的实现代码的文章就讲到这里了,如果你想了解更多关于spring boot 开发soap webservice的实现代码的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
背景: 我最近一直在使用 JPA,我为相当大的关系数据库项目生成持久层的轻松程度给我留下了深刻的印象。 我们公司使用大量非 SQL 数据库,特别是面向列的数据库。我对可能对这些数据库使用 JPA 有一
我已经在我的 maven pom 中添加了这些构建配置,因为我希望将 Apache Solr 依赖项与 Jar 捆绑在一起。否则我得到了 SolarServerException: ClassNotF
interface ITurtle { void Fight(); void EatPizza(); } interface ILeonardo : ITurtle {
我希望可用于 Java 的对象/关系映射 (ORM) 工具之一能够满足这些要求: 使用 JPA 或 native SQL 查询获取大量行并将其作为实体对象返回。 允许在行(实体)中进行迭代,并在对当前
好像没有,因为我有实现From for 的代码, 我可以转换 A到 B与 .into() , 但同样的事情不适用于 Vec .into()一个Vec . 要么我搞砸了阻止实现派生的事情,要么这不应该发
在 C# 中,如果 A 实现 IX 并且 B 继承自 A ,是否必然遵循 B 实现 IX?如果是,是因为 LSP 吗?之间有什么区别吗: 1. Interface IX; Class A : IX;
就目前而言,这个问题不适合我们的问答形式。我们希望答案得到事实、引用资料或专业知识的支持,但这个问题可能会引发辩论、争论、投票或扩展讨论。如果您觉得这个问题可以改进并可能重新打开,visit the
我正在阅读标准haskell库的(^)的实现代码: (^) :: (Num a, Integral b) => a -> b -> a x0 ^ y0 | y0 a -> b ->a expo x0
我将把国际象棋游戏表示为 C++ 结构。我认为,最好的选择是树结构(因为在每个深度我们都有几个可能的移动)。 这是一个好的方法吗? struct TreeElement{ SomeMoveType
我正在为用户名数据库实现字符串匹配算法。我的方法采用现有的用户名数据库和用户想要的新用户名,然后检查用户名是否已被占用。如果采用该方法,则该方法应该返回带有数据库中未采用的数字的用户名。 例子: “贾
我正在尝试实现 Breadth-first search algorithm , 为了找到两个顶点之间的最短距离。我开发了一个 Queue 对象来保存和检索对象,并且我有一个二维数组来保存两个给定顶点
我目前正在 ika 中开发我的 Python 游戏,它使用 python 2.5 我决定为 AI 使用 A* 寻路。然而,我发现它对我的需要来说太慢了(3-4 个敌人可能会落后于游戏,但我想供应 4-
我正在寻找 Kademlia 的开源实现C/C++ 中的分布式哈希表。它必须是轻量级和跨平台的(win/linux/mac)。 它必须能够将信息发布到 DHT 并检索它。 最佳答案 OpenDHT是
我在一本书中读到这一行:-“当我们要求 C++ 实现运行程序时,它会通过调用此函数来实现。” 而且我想知道“C++ 实现”是什么意思或具体是什么。帮忙!? 最佳答案 “C++ 实现”是指编译器加上链接
我正在尝试使用分支定界的 C++ 实现这个背包问题。此网站上有一个 Java 版本:Implementing branch and bound for knapsack 我试图让我的 C++ 版本打印
在很多情况下,我需要在 C# 中访问合适的哈希算法,从重写 GetHashCode 到对数据执行快速比较/查找。 我发现 FNV 哈希是一种非常简单/好/快速的哈希算法。但是,我从未见过 C# 实现的
目录 LRU缓存替换策略 核心思想 不适用场景 算法基本实现 算法优化
1. 绪论 在前面文章中提到 空间直角坐标系相互转换 ,测绘坐标转换时,一般涉及到的情况是:两个直角坐标系的小角度转换。这个就是我们经常在测绘数据处理中,WGS-84坐标系、54北京坐标系
在软件开发过程中,有时候我们需要定时地检查数据库中的数据,并在发现新增数据时触发一个动作。为了实现这个需求,我们在 .Net 7 下进行一次简单的演示. PeriodicTimer .
二分查找 二分查找算法,说白了就是在有序的数组里面给予一个存在数组里面的值key,然后将其先和数组中间的比较,如果key大于中间值,进行下一次mid后面的比较,直到找到相等的,就可以得到它的位置。
我是一名优秀的程序员,十分优秀!