java - 从 SOAP xml 解组,getSOAPBody() 方法返回 Null
问题描述
在解组 Soap 响应时,getSOAPBody 返回 null。我尝试使用在这种情况下有效的 XMLStreamReader。但我想知道一种通过 ByteArrayInputStream 做到这一点的方法。什么是正确的解决方法。我附在 POJO 和 package-info.java 文件下面。
此外,如果我将 "xsi:type="xsd:string" 设置为 "xsi:type="string" 它似乎可以工作。我应该如何修改我的 bean 类以使其正常工作?
包信息.java
@XmlSchema(
namespace = "http://tempuri.org/",
xmlns = {
@javax.xml.bind.annotation.XmlNs(prefix = "soap", namespaceURI = "http://schemas.xmlsoap.org/soap/envelope/"),
@javax.xml.bind.annotation.XmlNs(prefix = "xsi", namespaceURI = "http://www.w3.org/2001/XMLSchema-instance/"),
@javax.xml.bind.annotation.XmlNs(prefix = "xsd", namespaceURI = "http://www.w3.org/2001/XMLSchema")
})
package org.tempuri;
import javax.xml.bind.annotation.XmlSchema;
测试JAX.java
package org.test;
import java.io.ByteArrayInputStream;
import java.io.IOException;
import javax.xml.bind.JAXBContext;
import javax.xml.bind.JAXBException;
import javax.xml.bind.Unmarshaller;
import javax.xml.soap.MessageFactory;
import javax.xml.soap.SOAPException;
import javax.xml.soap.SOAPMessage;
import javax.xml.stream.XMLStreamException;
import org.tempuri.WIResponse;
/**
* Test Class
*
* @author vijay.prakash
*
*/
public class TestJAX {
public static void main(String[] args) throws XMLStreamException, JAXBException, IOException, SOAPException {
// xmlReadingWay
/*
* XMLInputFactory xif = XMLInputFactory.newFactory(); XMLStreamReader xsr =
* xif.createXMLStreamReader(new FileReader("D:\\BACKUPWARS\\some.xml"));
*
* xsr.nextTag(); // Advance to Envelope tag xsr.nextTag(); // Advance to Body
* tag xsr.nextTag(); // Advance to getNumberResponconse tag
*/
// byteway
String object = "<?xml version=\"1.0\" encoding=\"utf-8\"?>\r\n"
+ "<soap:Envelope xmlns:soap=\"http://schemas.xmlsoap.org/soap/envelope/\"\r\n"
+ " xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\" xmlns:xsd=\"http://www.w3.org/2001/XMLSchema\">\r\n"
+ " <soap:Body>\r\n" + " <WIResponse xmlns=\"http://tempuri.org/\">\r\n"
+ " <WIResult xsi:type=\"xsd:string\">20702458ghhc</WIResult>\r\n"
+ " </WIResponse>\r\n" + " </soap:Body>\r\n" + "</soap:Envelope>";
String str = new String(object.getBytes(), "utf-8");
final MessageFactory messFac = MessageFactory.newInstance();
SOAPMessage message = messFac.createMessage(null, new ByteArrayInputStream(str.getBytes()));
message.setProperty(SOAPMessage.CHARACTER_SET_ENCODING, "utf-8");
System.out.println("message: " + message.getSOAPBody());
JAXBContext jaxbContext = JAXBContext.newInstance(WIResponse.class);
Unmarshaller unmarshaller = jaxbContext.createUnmarshaller();
System.out.println("message.getSOAPBody(): " + message.getSOAPBody());
Object je = unmarshaller.unmarshal(message.getSOAPBody().extractContentAsDocument());
WIResponse wi = (WIResponse) je;
/*
* //JAXBElement<WIResponse> je =
* unmarshaller.unmarshal(message.getSOAPBody().extractContentAsDocument(),
* WIResponse.class);
*/
System.out.println(wi.getWIResult());
}
}
肥皂xml
<?xml version="1.0" encoding="utf-8"?>
<soap:Envelope xmlns:soap="http://schemas.xmlsoap.org/soap/envelope/"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema">
<soap:Body>
<WIResponse xmlns="http://tempuri.org/">
<WIResult xsi:type="xsd:string">20702458ghhc</WIResult>
</WIResponse>
</soap:Body>
</soap:Envelope>
WIResponse.java
package org.tempuri;
import javax.xml.bind.annotation.XmlAccessType;
import javax.xml.bind.annotation.XmlAccessorType;
import javax.xml.bind.annotation.XmlElement;
import javax.xml.bind.annotation.XmlRootElement;
@XmlAccessorType(XmlAccessType.FIELD)
@XmlRootElement(name = "WIResponse", namespace = "http://tempuri.org/")
public class WIResponse {
@XmlElement(name = "WIResult", required = true, type = java.lang.String.class)
String wIResult;
public String getWIResult() {
return wIResult;
}
public void setWIResult(String wIResult) {
this.wIResult = wIResult;
}
}
解决方案
该问题似乎是某种解析错误(getSOAPBody() 似乎无法正确处理命名空间解析)。
解决这个问题的“最简单”的方法是自己从 SOAP 消息中检索原始 XML,但是在典型的 Java 方式中,解决方案是复杂的和官僚的。
您可能需要修改下面的示例代码:
//https://docs.oracle.com/javase/7/docs/api/javax/xml/soap/SOAPPart.html
javax.xml.soap.SOAPPart GSP = SomeSOAPMessage.getSOAPPart();
//https://static.javadoc.io/com.sun.xml.messaging.saaj/saaj-impl/1.4.0/com/sun/xml/messaging/saaj/util/JAXMStreamSource.html
com.sun.xml.messaging.saaj.util.JAXMStreamSource GC = GSP.getContent();
//https://static.javadoc.io/com.sun.xml.messaging.saaj/saaj-impl/1.4.0/com/sun/xml/messaging/saaj/util/ByteInputStream.html
com.sun.xml.messaging.saaj.util.ByteInputStream BIS = GC.getInputStream();
byte[] ByteArray = BIS.getBytes();
java.lang.String Data = new java.lang.String(ByteArray, "UTF-8");
基本上,SOAPPart 从消息中获取 SOAPPart 的内容,它是一个 JAXMStreamSource。
然后从 JAXMStreamSource 中,它可以有一个“Reader”或一个“InputStream”(在我的例子中,我发现它有一个“InputStream”)。InputStream 可能由应用程序类型决定(在我的例子中,xop+xml,它是“二进制”XML),但是在测试中我发现它是 ByteInputStream 类型,它是 ByteArrayInputStream 的扩展类。
最后,您可以将 ByteArray 转换为字符串。此字符串将包含整个 XML(包括 XML 版本标头),然后您可以使用您想使用的任何首选 XML 处理程序类对其进行解析。
(包括全名解析和注释参考,以帮助理解其工作原理。)
推荐阅读
- c# - 使用 .Net Core 2.0 编写的 AWS Lambda 函数无法加载 libdl
- c - 在 VS 2017 中禁用 AVX-512 内部函数
- java - 链 CompletableFuture 并在第一次成功时停止
- c# - crud 更新无法正常工作
- python - 如何将变量设置为问题总数
- css - REM 是基于默认浏览器/操作系统字体的默认字体大小、自定义提供的字体还是实际固定像素呈现?
- python - 使用 ctypes 将十六进制值从 C Dll 返回到 Python
- mysql - MySQL左连接计数未显示左表中的所有条目
- python - 坚持一个大 scipy.sparse.csr_matrix
- node.js - Heroku + MERN 堆栈 - 任何值的 HTTP POST 请求无效