从Android kSOAP反序列化WCF服务中的参数时出错

时间:2011-08-02 03:48:08

标签: android wcf soap deserialization ksoap2

8/5/11:在帖子底部附近更新

我正在尝试使用kSOAP从我的Android应用调用我的WCF Web服务方法。该方法需要4个参数。其中3个是Guid类型(为编组Java的UUID工作),最后一个是自定义类型:LocationLibrary.Location。这个类型位于一个单独的DLL(LocationLibrary)中,我加载到WCF Web服务项目中,它基本上由两个双精度数组成,纬度和经度。

    [OperationContract]       
    byte[] GetData(Guid deviceId, Guid appId, Guid devKey, LocationLibrary.Location loc);

项目LocationLibrary中的Location类非常简单:

    namespace LocationLibrary
    {
    public class Location
    {
        public double latitude { get; set; }

        public double longitude { get; set; }

        public new string ToString()
        {
            return latitude.ToString() + "," + longitude.ToString();
        }

        public bool Equals(Location loc)
        {
            return this.latitude == loc.latitude && this.longitude == loc.longitude;
        }
    }
}

在我的Android项目中,我创建了一个名为“Location”的类,类似于.NET版本:

    public class Location {
    public double latitude;
    public double longitude;

    public Location() {}

    public static Location fromString(String s)
    {
    //Format from SOAP message is "anyType{latitude=39.6572799682617; longitude=-78.9278602600098; }"
    Location result = new Location();
    String[] tokens = s.split("=");
    String lat = tokens[1].split(";")[0];
    String lng = tokens[2].split(";")[0];
    result.latitude = Double.parseDouble(lat);
    result.longitude = Double.parseDouble(lng); 
    return result;
    }

    public String toString()
    {       
    return Double.toString(latitude) + "," + Double.toString(longitude);    
    }
}

使用kSOAP连接到Web服务时,我会执行以下操作:

    private final String SOAP_ACTION = "http://tempuri.org/IMagicSauceV3/GetData";
    private final String OPERATION_NAME = "GetData";     
    private final String WSDL_TARGET_NAMESPACE = "http://tempuri.org/";
    private final String SOAP_ADDRESS = "http://mydomain.com/myservice.svc";
    private final SoapSerializationEnvelope envelope;

    SoapObject request = new SoapObject(WSDL_TARGET_NAMESPACE, OPERATION_NAME);
    request.addProperty(Cloud8Connector.deviceIdProperty);
    request.addProperty(Cloud8Connector.appIdProperty);
    request.addProperty(Cloud8Connector.devKeyProperty);        
    PropertyInfo locationProperty = new PropertyInfo();
    locationProperty.setName("loc");
    locationProperty.setValue(Cloud8Connector.myLoc);
    locationProperty.setType(Location.class);
    request.addProperty(locationProperty);

    envelope = new SoapSerializationEnvelope(SoapEnvelope.VER11);
    envelope.dotNet = true;
       envelope.setOutputSoapObject(request);
       MarshalUUID mu = new MarshalUUID();
    mu.register(envelope);
    MarshalLocation ml = new MarshalLocation();
    ml.register(envelope);

     byte[] result = null;
    HttpTransportSE httpRequest = new HttpTransportSE(SOAP_ADDRESS);

    try
    {
        httpRequest.call(SOAP_ACTION, envelope);

        String payloadString = ((SoapPrimitive)(envelope.getResponse())).toString();
        result = Base64.decode(payloadString, Base64.DEFAULT);         
    }
    catch(Exception e)
    {
        e.printStackTrace();
    }

如您所见,我为Location创建了一个简单的Marshal类,它基本上只使用fromString和toString方法。我使用位置:

的编组实例注册信封
    MarshalLocation ml = new MarshalLocation();
    ml.register(envelope);

这是位置的编组类:

    public class MarshalLocation implements Marshal
    {

    public Object readInstance(XmlPullParser parser, String namespace, String name, 
            PropertyInfo expected) throws IOException, XmlPullParserException {
        return Location.fromString(parser.nextText());                                
    }

    public void register(SoapSerializationEnvelope cm) {
         cm.addMapping(cm.xsd, "Location", Location.class, this);

    }

    public void writeInstance(XmlSerializer writer, Object obj) throws IOException {

            writer.text(((Location)obj).toString());             
        }    
    }

但是,我从WCF返回此错误,似乎无法使其工作。从错误和搜索来看,我认为我需要使用我的Web服务调整一些内容,但我不确定究竟什么是解决此问题的最佳方法。

我已经尝试将此添加到Web服务的web.config文件中:

        <system.runtime.serialization>
    <dataContractSerializer>
        <declaredTypes>
            <add type="Location, LocationLibrary, Version=1.0.0.0,Culture=neutral,PublicKeyToken=null">
                <knownType type="Location, LocationLibrary, Version=1.0.0.0,Culture=neutral,PublicKeyToken=null"></knownType>
            </add>
        </declaredTypes>
    </dataContractSerializer>
</system.runtime.serialization>

我尝试将ServiceKnownType属性添加到接口方法签名中:

    [OperationContract]       
    [ServiceKnownType(typeof(LocationLibrary.Location))]
    byte[] GetData(Guid deviceId, Guid appId, Guid devKey, LocationLibrary.Location loc);

但我仍然得到这个错误:(

你能指出我正确的方向吗?谢谢!

错误:

  

07-30 00:42:08.186:WARN / System.err(8723):SoapFault - faultcode:   'a:DeserializationFailed'faultstring:'格式化程序抛出了一个   尝试反序列化消息时出现异常:出现错误   在尝试反序列化参数http://tempuri.org/:loc时。该   InnerException消息是'第1行位置522中的错误。元素   'http://tempuri.org/:loc'包含映射到的类型的数据   名称为“http://www.w3.org/2001/XMLSchema:Location”。解串器有   不知道映射到此名称的任何类型。考虑使用   DataContractResolver或添加与“Location”对应的类型   已知类型的列表 - 例如,通过使用KnownTypeAttribute   属性或将其添加到传递给的已知类型列表中   DataContractSerializer的。“。有关更多详细信息,请参阅InnerException。   faultactor:'null'详细信息:org.kxml2.kdom.Node@4053aa28

更新

理想情况下,我不必在主WCF项目中移动LocationLibrary类,因为这会使在不同命名空间中期望它的旧客户端的处理变得复杂。但是,我确实通过在主WCF项目中移动这些类并将java中的marshal类修改为以下内容来实现工作:

    public void register(SoapSerializationEnvelope cm) {            
         cm.addMapping("http://schemas.datacontract.org/2004/07/MagicSauceV3", "Location", Location.class, this);
    }        

    public void writeInstance(XmlSerializer writer, Object obj) throws IOException {
            Location loc = (Location)obj;
            writer.startTag("http://schemas.datacontract.org/2004/07/MagicSauceV3", "latitude");
            writer.text(Double.toString(loc.latitude));
            writer.endTag("http://schemas.datacontract.org/2004/07/MagicSauceV3", "latitude");
            writer.startTag("http://schemas.datacontract.org/2004/07/MagicSauceV3", "longitude");
            writer.text(Double.toString(loc.longitude));
            writer.endTag("http://schemas.datacontract.org/2004/07/MagicSauceV3", "longitude");       
        }

现在我已经开始工作,我只是使用LocationLibrary WCF类来处理它们(在一个单独的DLL中)。所以我修改了marshal类:

    public void register(SoapSerializationEnvelope cm) {            
         cm.addMapping("http://schemas.datacontract.org/2004/07/LocationLibrary", "Location", Location.class, this);
    }


    public void writeInstance(XmlSerializer writer, Object obj) throws IOException {
            Location loc = (Location)obj;
            writer.startTag("http://schemas.datacontract.org/2004/07/LocationLibrary", "latitude");
            writer.text(Double.toString(loc.latitude));
            writer.endTag("http://schemas.datacontract.org/2004/07/LocationLibrary", "latitude");
            writer.startTag("http://schemas.datacontract.org/2004/07/LocationLibrary", "longitude");
            writer.text(Double.toString(loc.longitude));
            writer.endTag("http://schemas.datacontract.org/2004/07/LocationLibrary", "longitude");       
        }

这看起来应该会成功,因为它会生成与WP7版本类似的XML。

WP7工作XML请求:

<GetData xmlns="http://tempuri.org/">
<deviceId>{valid GUID}</deviceId>
<appId>{valid GUID}</appId>
<devKey>{valid GUID}</devKey>
<loc xmlns:d4p1="http://schemas.datacontract.org/2004/07/LocationLibrary" xmlns:i="http://www.w3.org/2001/XMLSchema-instance">
<d4p1:latitude>47.65</d4p1:latitude>
<d4p1:longitude>-122.34</d4p1:longitude>
</loc>
</GetData>

Android非工作XML请求:

<GetData xmlns="http://tempuri.org/" id="o0" c:root="1">
<deviceId i:type="d:UUID">{valid GUID}</deviceId>
<appId i:type="d:UUID">{valid GUID}</appId>
<devKey i:type="d:UUID">{valid GUID}</devKey>
<loc i:type="n0:Location" xmlns:n0="http://schemas.datacontract.org/2004/07/LocationLibrary">
<n0:latitude>47.65</n0:latitude>
<n0:longitude>-122.34</n0:longitude>
</loc>
</GetData>

上述Android XML请求会产生此响应错误:

08-05 22:51:23.703:WARN / System.err(1382):SoapFault - faultcode:'a:DeserializationFailed'faultstring:'格式化程序在尝试反序列化消息时抛出异常:出现错误尝试反序列化参数http://tempuri.org/:loc。 InnerException消息是'第1行位置590中的错误。元素'http://tempuri.org/:loc'包含映射到名称'http://schemas.datacontract.org/2004/07/的类型的数据LocationLibrary:位置”。反序列化器不知道映射到此名称的任何类型。考虑使用DataContractResolver或将与“Location”对应的类型添加到已知类型列表中 - 例如,通过使用KnownTypeAttribute属性或将其添加到传递给DataContractSerializer的已知类型列表中。有关更多详细信息,请参阅InnerException。 faultactor:'null'详细信息:org.kxml2.kdom.Node@4054bcb8

我看到的唯一值得注意的差异是工作XML请求中“loc”的属性:          的xmlns:I = “http://www.w3.org/2001/XMLSchema-instance”

我不知道这是否会产生影响(并且它似乎没有被嵌套标签引用)。我也不知道如何添加额外的命名空间而不会弄乱另一个。

对于长篇文章感到抱歉,但我想确保您拥有所需的所有帮助信息:)

谢谢!

2 个答案:

答案 0 :(得分:1)

正如我所料,你只是做ToString,但这是错误的。您的位置不会像两个连接值的字符串一样传输。它作为序列化的对象传输到XML。类似的东西:

<Location> <!-- or loc -->
   <latitude>10.0</latitude>
   <longitude>-20.0</longitude>
</Location>

答案 1 :(得分:1)

感谢Ladislav的正确方向!我不确定这里的礼仪是什么奖励,但如果有人会建议,我很乐意跟进。

答案是双重的,我可以确认它现在可以在一个单独的DLL中使用LocationLibrary的Location类:

  1. 使用我的编组课改变我在Android中形成XML请求的方式。 writeInstance方法只使用XmlSerializer类的.textMethod,在Location类上调用.toString()。我改变它以利用形成XML标记的正确方法(.startTag和.endTag方法):

    public void writeInstance(XmlSerializer writer, Object obj) throws IOException {
        Location loc = (Location)obj;
        writer.startTag("http://schemas.datacontract.org/2004/07/LocationLibrary", "latitude");
        writer.text(Double.toString(loc.latitude));
        writer.endTag("http://schemas.datacontract.org/2004/07/LocationLibrary", "latitude");
        writer.startTag("http://schemas.datacontract.org/2004/07/LocationLibrary", "longitude");
        writer.text(Double.toString(loc.longitude));
        writer.endTag("http://schemas.datacontract.org/2004/07/LocationLibrary", "longitude");       
    }
    
  2. 向WCF服务添加一些简单标记。我将[DataContract]添加到WCF服务使用的LocationLibrary程序集中的Location类,并将[DataMember]添加到其每个成员中:

    [DataContract]
    public class Location
    {
    [DataMember]
    public double latitude { get; set; }
    
    [DataMember]
    public double longitude { get; set; }
    
    public new string ToString()
    {
        return latitude.ToString() + "," + longitude.ToString();
    }
    
    public bool Equals(Location loc)
    {
        return this.latitude == loc.latitude && this.longitude == loc.longitude;
    }