San*_*man 5 c# wcf json known-types deserialization
我创建了以下类.一个基类IObject和2个派生类A和B.
[KnownType(typeof(B))]
[KnownType(typeof(A))]
[DataContract(Name = "IObject")]
public class IObject
{
}
[DataContract(Name="A")]
public class A : IObject
{
[DataMember]
public string s1 { get; set; } // Tag Name as it will be presented to the user
}
[DataContract(Name="B")]
public class B : IObject
{
[DataMember]
public string s2 { get; set; }
}
Run Code Online (Sandbox Code Playgroud)
我还创建了以下服务:
[ServiceKnownType(typeof(B))]
[ServiceKnownType(typeof(A))]
public void GetR(IObject obj)
{
B other = (B)obj;
}
Run Code Online (Sandbox Code Playgroud)
我想要做的是得到一个A或B的实例,但我不知道我将得到哪一个,所以我希望得到一个IObject,然后将其转换为A或B,如我所放置的示例所示.
当我发送包含s2字符串的json字符串时会发生什么,我得到IObject实例而不是B实例.代码有什么问题?
我正在使用的客户端示例:
var url = serviceURL;
var data = {s2: "Data"};
var json = JSON.stringify(data);
$.ajax({
type: "POST",
url: url,
data: data,
contentType: "application/json",
dataType: 'json'
});
Run Code Online (Sandbox Code Playgroud)
编辑:我已经通过以下链接将示例代码上传到gitHub:https: //github.com/AlgoEitan/27588144
客户端有ac#client(我用c#和javascript尝试过它 - web浏览器客户端)
DataContractJsonSerializer有一种特定的格式,它存储有关多态类型的已知类型信息的提示,可以通过一些测试发现这些提示。我按原样复制并粘贴了您的类,并创建了以下测试代码:
public static class DataContractJsonSerializerPolymorphismTest
{
public static void Test()
{
var a1 = new A() { s1 = "A" };
var b1 = new B() { s2 = "B" };
DataContractJsonSerializer serializer = new DataContractJsonSerializer(typeof(IObject));
var jsona1 = DataContractJsonSerializerHelper.GetJson(a1, serializer);
var jsonb1 = DataContractJsonSerializerHelper.GetJson(b1, serializer);
Debug.WriteLine(jsona1);
Debug.WriteLine(jsonb1);
var newa1 = DataContractJsonSerializerHelper.GetObject<IObject>(jsona1, serializer);
Debug.Assert(newa1.GetType() == a1.GetType()); // No assert
}
}
Run Code Online (Sandbox Code Playgroud)
这样就创建了以下 JSON:
{"__type":"A:#Tile.DataContractJsonSerializerPolymorphism","s1":"A"}
{"__type":"B:#Tile.DataContractJsonSerializerPolymorphism","s2":"B"}
Run Code Online (Sandbox Code Playgroud)
其中Tile.DataContractJsonSerializerPolymorphism恰好是我的测试应用程序中 CLR 命名空间的名称。因此,如您所见,已知的类型信息隐藏在这个额外的 JSON 属性中__type。DataContractJsonSerializerHelper现在,如果您也在客户端中使用 ,您将永远不会知道这一点,因为通信会正常工作。但你使用的JSON.stringify()which没有这个逻辑。因此,您可能必须"__type":"DataContractName:DataContractNamespace"在客户端手动添加该属性。
有关多态类型格式的更多信息可以在此处找到。(我只是在找到隐藏参数后才追踪到这个文档__type,这给了我一个额外的谷歌搜索词。)
仅供参考,这是我在测试代码中使用的帮助程序类:
public static class DataContractJsonSerializerHelper
{
private static MemoryStream GenerateStreamFromString(string value)
{
return new MemoryStream(Encoding.Unicode.GetBytes(value ?? ""));
}
public static string GetJson<T>(T obj, DataContractJsonSerializer serializer) where T : class
{
using (var memory = new MemoryStream())
{
serializer.WriteObject(memory, obj);
memory.Seek(0, SeekOrigin.Begin);
using (var reader = new StreamReader(memory))
{
return reader.ReadToEnd();
}
}
}
public static string GetJson<T>(T obj) where T : class
{
DataContractJsonSerializer serializer = new DataContractJsonSerializer(typeof(T));
return GetJson(obj, serializer);
}
public static T GetObject<T>(string json) where T : class
{
DataContractJsonSerializer serializer = new DataContractJsonSerializer(typeof(T));
return GetObject<T>(json, serializer);
}
public static T GetObject<T>(string json, DataContractJsonSerializer serializer) where T : class
{
T obj = null;
using (var stream = GenerateStreamFromString(json))
{
obj = (T)serializer.ReadObject(stream);
}
return obj;
}
}
Run Code Online (Sandbox Code Playgroud)
| 归档时间: |
|
| 查看次数: |
266 次 |
| 最近记录: |