我在一次采访中遇到过这个问题,请问SOAP Web 服务是否只支持“POST”http 方法,或者是否有某种方法可以在服务器端接受其他方法?
我一直使用 POST,但根据 W3C standard,SOAP 支持 POST 和 GET 方法。
编辑:经过一些研究,这似乎并不完全正确,如您所见 here。 理论上可以使用 GET,因为 POST 和 GET 是 HTTP 传输协议的方法,而 SOAP 可以通过 HTTP 使用。
但如您所知,GET 将请求包含在查询字符串中。 SOAP 请求(XML 消息)通常过于复杂和冗长,无法包含在查询字符串中,因此几乎每个实现(例如 JAX-WS)都只支持 POST。
Thread 已经成立三年了,但我认为仍有很多人会向自己提出同样的问题,并会在网络上找到错误的答案。问题的答案是否定的,也可以使用 GET 方法。
根据 SOAP 规范,可在此处找到:https://www.w3.org/TR/2007/REC-soap12-part0-20070427/#transport GET 和 POST 方法都可用于通过 http 交换 SOAP 消息。使用 HTTP POST 方法在 HTTP 请求正文中传送 SOAP 消息使用了一种称为 SOAP 请求-响应消息交换模式的模式。在 HTTP GET 的情况下,使用称为 SOAP 响应消息交换模式的模式。这两种模式的主要区别是:
第一种类型的交互允许使用 HTTP POST 正文中的数据来创建或修改由 HTTP 请求所指向的 URI 标识的资源的状态。第二种类型的交互模式提供了使用 HTTP GET 请求获取资源表示的能力,而无需以任何方式改变其状态。在第一种情况下,特定于 SOAP 的关注点是 HTTP POST 请求的主体是一个 SOAP 消息,它必须被处理(根据 SOAP 处理模型)作为特定于应用程序的处理的一部分,以符合POST 语义。在第二种情况下,可以预见的典型用法是,所请求资源的表示不是作为 HTML 或实际上是通用 XML 文档返回,而是作为 SOAP 消息返回的情况。也就是说,响应消息的 HTTP 内容类型标头将其标识为媒体类型“application/soap+xml”
所以 GET 和 POST 方法都可以使用。另一件事是在实践中主要使用 POST 方法。
坏事是当比较 RESTful 服务和 SOAP 服务时,作为 REST 的一个优势,人们带来了缓存,这在 SOAP 中不可用,因为 SOAP 只使用 POST。这是完全错误的。
这是 SOAP 中的 GET 实现:
@WebServiceProvider(targetNamespace="http://attachment.service.soap.com/download")
@ServiceMode(value = javax.xml.ws.Service.Mode.MESSAGE)
@BindingType(value = HTTPBinding.HTTP_BINDING)
public final class ImageDownloadServiceProvider implements Provider<DataSource> {
@Resource
private WebServiceContext wsContext;
@Override
public DataSource invoke(DataSource request) {
if (wsContext == null)
throw new RuntimeException("dependency injection failed on wsContext");
MessageContext msgContext = wsContext.getMessageContext();
HttpExchange exchange = (HttpExchange) msgContext.get("com.sun.xml.internal.ws.http.exchange");
String filename = exchange.getRequestURI().getQuery().replace("file=", "");
switch ((String) msgContext.get(MessageContext.HTTP_REQUEST_METHOD)) {
case "GET":
return doGet(filename);
default:
throw new HTTPException(405);
}
}
private DataSource doGet(String filename) {
FileDataSource fds = new FileDataSource(filename);
MimetypesFileTypeMap mtftm = new MimetypesFileTypeMap();
mtftm.addMimeTypes("image/jpeg jpg");
fds.setFileTypeMap(mtftm);
return fds;
}