发生在我身上的实际故事,最后发现和axis解析xml时的处理机制有关,namespace的有无会影响xml解析的方式,简单的说就是有namespace按照元素名解析,没有namespace则按照index下标的顺序来解析。
中间惊险,一一道来,做技术的不容易啊。
这个市公司的一个大项目,使用web service,我负责服务器端的开发,其他厂商开发客户端。好说,axis上,几个月下来,设计/开发/测试一路ok,就进移动研究院准备最后的入网测试了。
和我们一起联合测试的cx公司,报告说发现错误,经查找是服务器端解析他们发过来的请求失败,出现异常 java.lang.NumberFormatException。非常郁闷,按说不大可能,代码是从wsdl文件自动生成的,怎么可能出这种低级错误,而且我们自己反复测试都通过。于是想办法抓包,发现他们的报文如下:
<env:Envelope xmlns:env='http://schemas.xmlsoap.org/soap/envelope/'>
<env:Header/>
<env:Body>
<ssoRegister>
<Version>0100</Version>
<OpCode>D0001</OpCode>
<UID>13689000001</UID>
<UIDType>SYSUSER</UIDType>
<Service>WEBADMIN</Service>
<LocalZone>5</LocalZone>
<Province>0</Province>
<Privilege>SUPER</Privilege>
</ssoRegister>
</env:Body>
</env:Envelope>
格式怪怪的,对照标准的报文:
<?xml version="1.0" encoding="UTF-8"?>
<soapenv:Envelope xmlns:soapenv="http://schemas.xmlsoap.org/soap/envelope/" xmlns:soapenc="http://schemas.xmlsoap.org/soap/en
coding/" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
<soapenv:Body>
<ssoRegister xmlns="http://system.chinamobile.com">
<OpCode>D0001</OpCode>
<UID>13660000001</UID>
<UIDType>MDN</UIDType>
<Service>WEBMAIL</Service>
<LocalZone>0</LocalZone>
<Province>0</Province>
<Privilege>SUPER</Privilege>
</ssoRegister>
</soapenv:Body>
</soapenv:Envelope>
发现他们的请求多了一个 <Version>0100</Version>字段,试着去掉,解析就通过了。于是通知cx公司修改,同时提示他们说他们的格式不够标准,xsd的namespace很多都没有写。当时就奇怪,从现象看似乎axis是按照index/下标顺序来解析xml,因为多了一个 version,因此后面的字段都顺推了一位,造成用"WEBMAIL"的值来作为LocalZone解析,而localzone是int类型,所以解析失败,出现异常:java.lang.NumberFormatException.
有些奇怪axis怎么会这样解析xml,当时忙也无暇细想。继续测试中又发现另外一个接口出现问题,cx公司的soap请求的xml字段顺序和wsdl文件规定的顺序不一致,造成数据解析出来内容错乱。晕倒,细问才知道cx公司不是按照wsdl来自动生成代码,也不依照wsdl文件的格式要求,而是很奇怪的以协议附件中的soap请求示例为基准(很荒谬的事情,这还是电信级别的软件开发方式,想不通)。偏偏协议在一个接口上wsdl和示例的顺序不一致,造成这个问题。
之后就是非技术的扯皮了,总之cx公司坚持他们的正确性和合理性,非逼我们公司修改服务器端做法。细的不说了,俺们技术人员不懂也不该去关注,黑暗的内幕。由于那个协议的内容是俺修改的(前人离职了),这个出问题的地方就是我陆续修订的,于是责任就压到我身上了,当时那个郁闷啊。写了封邮件准备给领导,将事情说清楚,认错并承认是俺的责任......惨就一个字。就在邮件发出去之后,突然想到,恩,怎么老是按照index解析呢,axis没有这么笨吧?用测试脚本又测试了一下,没有问题,再看soap 报文,惊奇的发现顺序也是有差异的,但是怎么服务器端就能正确解析呢?
灵光一动,想起cx的报文格式来了,他们的格式非常的不规范,简直就粗糙到极点了,当时我们几个研发还说笑,说他们肯定是手工拼凑文本,将 soap/web service退化为http + xml,然后再将xml退化为文本。难道是格式的问题?对照了一下,发现少了<*** xmlns="http://system.chinamobile.com">这里的namespace,试着将cx的报文加上这个 namespace,然后用脚本工具提交测试,服务器端解析ok。
这下问题明朗了,可以发现是这样的规律,axis在解析时发现没有namespace,就按照顺序来解析:
wsdl标准顺序 实际报文顺序 最终解析出来内容
<ServiceCode> <Alias></Alias> --〉 serviceCode=
<Source> <Source>WEB</Source> --〉 source=WEB
<Alias> <ServiceCode>YXZZY</ServiceCode> --〉 alias=YXZZY
于是长出一口气,又赶紧写了封新邮件,解释清楚终于将责任踢给cx了.真是惊险。平时哪里会遇到这样格式不规范的报文,这次长见识了.cx终于不再坚持了,增加了namespace后测试通过,最后赶在移动的时间期限前完成了测试,大家不用互相推责任了。
分享到:
相关推荐
总结,处理SOAP XML报文中的节点名首字母转换是一个涉及XML解析、DOM操作和递归过程的编程任务。在Axis2中,我们可以利用OMElement的API实现这一目标,从而满足特定的规范或需求。理解这些概念对于进行Web服务开发和...
同时,必须对可能出现的网络异常、解析异常等进行处理,确保应用的健壮性。 总结,Android环境下使用Ksoap2连接Axis2服务,需要理解SOAP协议、Ksoap2库的工作原理,以及如何处理自定义类对象和XML响应。通过以上...
通过对SOAP请求的解析和响应的生成,我们可以实现对特定数据的处理。此外,通过直接操作系统目录级的方式,可以更加灵活地管理Web服务。整个过程涉及到了Axis2的部署、Web服务的创建、SOAP请求和响应的处理等方面的...
Axis作为一个Web服务栈,负责解析和处理这些协议。 创建Web服务的第一步是准备Java类。假设我们有一个名为`HelloWorld`的服务类,它有一个`sayHello`方法,用于返回一个问候语: ```java public class HelloWorld ...
根据服务定义,这可能涉及解析XML响应,提取所需的数据。 6. **错误处理**:别忘了添加适当的错误处理机制,以处理可能出现的网络问题、服务不可用或无效SOAP消息等问题。 在实际应用中,你可能需要处理更复杂的...
3. 发送SOAP请求到webService并解析返回的结果。 4. 处理返回数据,展示或使用结果。 例如,创建一个Android方法来调用webService的`plus`方法: ```java private String callWebService(float x, float y) { ...
在Android客户端,我们使用Ksoap2库来构建SOAP请求和解析响应。这个库简化了SOAP通信的复杂性,使得开发者可以更专注于业务逻辑而不是底层的通信细节。 ```java public static List<HashMap> doCallWebService...
import javax.xml.namespace.QName; import org.apache.axiom.om.OMAbstractFactory; import org.apache.axiom.om.OMElement; import org.apache.axiom.om.OMFactory; import org.apache.axiom.om.OMNamespace;...
5. **处理响应**:根据Web Service返回的数据类型,解析XML或JSON,将数据转化为对应的Java对象,然后在主线程中更新UI。 6. **异常处理**:在网络通信中,应处理可能的异常,如SocketTimeoutException、...
总结来说,Java调用WebService接口主要通过解析WSDL文件来了解服务的接口定义,然后使用相应的客户端库(如Apache Axis或JAX-WS)构建请求并发送。这些库通常会处理底层的SOAP消息构建和HTTP传输,让开发者可以专注...
import javax.xml.namespace.QName; import java.lang.Integer; import javax.xml.rpc.ParameterMode; public class caClient { public static void main(String[] args) { try { // 设置目标Web服务的地址 ...
在Java中,我们主要使用Apache的Axis库或者JAX-WS(Java API for XML Web Services)来处理SOAP请求。对于Android,由于其资源限制,我们通常使用Ksoap2库,这是一个轻量级的库,适合在移动设备上使用。 步骤1:...
}` 这里是对可能出现的异常进行捕获和处理,通常包括网络错误、解析错误等。 9. **不同Web服务的调用方式**: - `webService0`和`webService1`展示了如何调用不同的Web服务。虽然基本步骤相似,但每个服务可能有...
- **轴(Axis)**:轴定义了节点之间的关系,例如子节点轴(child axis)、父节点轴(parent axis)、属性轴(attribute axis)等。 - **函数与运算符**:XPath提供了丰富的函数库,如`count()`、`contains()`、`...
### 初探WSDL2JAVA工具的使用:深入解析与实践指南 在现代软件开发领域,Web服务(WebService)已成为企业级应用间进行交互的重要手段。Web服务定义语言(WSDL,Web Service Definition Language)是一种基于XML的...
请注意,实际应用中,你可能需要处理错误、解析返回的XML响应、以及可能的异常情况。此外,如果Web服务位于安全的HTTPS协议上,还需要处理证书验证等问题。 总的来说,通过MSXML调用Web服务涉及的关键步骤包括:...
SOAP(Simple Object Access Protocol)是一种基于XML的协议,用于在Web服务中交换结构化和类型化的信息。SOAP消息格式是这种通信的基础,它定义了如何包装数据以便在不同的应用程序之间进行传输。这篇博客文章...
ksoap2是一个用于Android平台的开源SOAP客户端库,它可以简化SOAP请求的创建和解析过程。ksoap2支持多种协议版本,如SOAP 1.1和SOAP 1.2,并且提供了易于使用的API,使得开发者能够轻松地与WebService进行交互。 ##...
在Android和SQL Server之间,我们通常使用SOAP(Simple Object Access Protocol)Web服务,它是一种基于XML的消息传递协议,适合于结构化的数据交换。 1. **设置开发环境**: - 安装Android Studio和配置Android ...
在早期的Web服务开发中,开发者通常会使用CXF、Axis2、XFire等工具来构建基于Java的应用程序。这些工具之所以被广泛采用,主要是因为它们能够提供广泛的兼容性和灵活性,并且适应多种不同的Java版本环境。然而,在...