最近在做CRM项目时,使用C#调用SAP PI发布的WebService服务时遇到的问题: 向WebService传递decimal、double、int、DateTime等非string类型数据时,服务器端接收不到数据。查询了很多资料,终于解决了问题,总结如下。
问题现象:
用C#.NET调用PI开发的WebService时,在客户端将封装的带有decimal属性的对象传到服务器端时,服务器端可以得到string类型的属性值,却不能得到int、double、decimal类型和DateTime类型的属性值
问题结症:
简单说一下WebService的工作原理。客户端调用一个WebService的方法,首先需要将方法名和需要传递的参数包装成XML(也就是SOAP包),通常是通过HTTP传递到服务器端,然后服务器端解析这段XML,得到被调用方法名称和传递来的参数,进而调用WebService实例的相应方法。方法执行完成之后,将返回的结果再包装成XML(SOAP响应)发送到客户端,客户端解析这段XML,进而得到返回结果。这里关键的地方在于中间加入了对象和XML相互转换的过程。
由于服务端对象的非String属性(int,double,decimal,DateTime...)设置了可以为空(minOccurs="0"),在客户端调用服务方法SI_Create(I_Customer customer )的时候传递一个Customer对象,设置了可以为空的非String属性在服务端接收不到属性数据。
以decimal为例 :
服务端程序:SI_Customer_Out_SynService对外提供SI_Create(I_Customer customer )方法,返回Customer编号,其中Customer包含属性: UMSA1(decimal类型)。
问题解决:
1.Web服务端配置
<xsd:element name="UMSA1" minOccurs="0"> <xsd:annotation> <xsd:documentation>Annual sales</xsd:documentation> </xsd:annotation> <xsd:simpleType> <xsd:restriction base="xsd:decimal"> <xsd:totalDigits value="15"/> <xsd:fractionDigits value="2"/> </xsd:restriction> </xsd:simpleType> </xsd:element>
2.在客户端调用时进行设置
VS2013封装WebService引用时,如果WebService发布的是一个基本数据类型组成的对象,则进行如下设置:
a. 该对象中非string属性为minOccurs="0" 设置时(表示该属性可以为空)),则会对该对象的非string属性同时生成两个属性,一般为"属性名Field"、"属性名Specified"。而"属性名Specified"是一个bool类型,只有这个属性被设置成true时,"属性Field"的值才会被序列化成xml传递。否则,服务端接收不到属性值。
b. 该对象中非string属性为minOccurs="1" maxOccurs="1"设置时(表示该属性不能为空,为必填项),如果该属性不赋值,则服务端则接收到的值为0.00。
参考:http://liyuandong.iteye.com/blog/782246