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”
我不知道这是否会产生影响(并且它似乎没有被嵌套标签引用)。我也不知道如何添加额外的命名空间而不会弄乱另一个。
对于长篇文章感到抱歉,但我想确保您拥有所需的所有帮助信息:)
谢谢!
答案 0 :(得分:1)
正如我所料,你只是做ToString
,但这是错误的。您的位置不会像两个连接值的字符串一样传输。它作为序列化的对象传输到XML。类似的东西:
<Location> <!-- or loc -->
<latitude>10.0</latitude>
<longitude>-20.0</longitude>
</Location>
答案 1 :(得分:1)
感谢Ladislav的正确方向!我不确定这里的礼仪是什么奖励,但如果有人会建议,我很乐意跟进。
答案是双重的,我可以确认它现在可以在一个单独的DLL中使用LocationLibrary的Location类:
使用我的编组课改变我在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");
}
向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;
}