【问题标题】:Deserialize a List<AbstractClass> with newtonsoft.json使用 newtonsoft.json 反序列化 List<AbstractClass>
【发布时间】:2013-11-18 01:10:21
【问题描述】:

我正在尝试序列化和反序列化abstract 类的列表(mustinherit 用于 vb),显然其中只有派生类的实例。

我用JsonProperty(ItemTypeNameHandling = TypeNameHandling.Auto) 修饰了列表参数,获得如下输出:

但是当我反序列化它时一直说他不能反序列化抽象类。

http://james.newtonking.com/json/help/index.html?topic=html/SerializeTypeNameHandling.htm

public class ConcreteClass
{
    private ObservableCollection<AbstractClass> _Nodes = new ObservableCollection<AbstractClass>();
    //<Newtonsoft.Json.JsonProperty(itemtypenamehandling:=Newtonsoft.Json.TypeNameHandling.Auto)>
    public ObservableCollection<AbstractClass> Nodes {
        get { return this._Nodes; }
    }
    public string Name { get; set; }
    public int Id { get; set; }
}

public abstract class AbstractClass
{
    private ObservableCollection<AbstractClass> _Nodes = new ObservableCollection<AbstractClass>();
    [Newtonsoft.Json.JsonProperty(itemtypenamehandling = Newtonsoft.Json.TypeNameHandling.Auto)]
    public ObservableCollection<AbstractClass> Nodes {
        get { return this._Nodes; }
    }
}

删除它的注释行!

【问题讨论】:

  • 请发布您的反序列化代码。
  • 另外,请发布您的课程。
  • 我一到办公室就会更新问题

标签: c# .net vb.net serialization json.net


【解决方案1】:

确保在反序列化时指定 TypeNameHandling,根据文档:

// for security TypeNameHandling is required when deserializing
Stockholder newStockholder = JsonConvert.DeserializeObject<Stockholder>(jsonTypeNameAuto, new JsonSerializerSettings
{
    TypeNameHandling = TypeNameHandling.Auto
});

值得注意的是,文档正在反序列化一个包含抽象类集合的具体类。

作为一个实验,尝试创建一个一次性类(具体),该类具有一个包含抽象对象列表的单一属性,并查看是否可以对其进行序列化和反序列化。

更新:

我刚刚在 LINQPad 中测试了以下代码:

void Main()
{
    var test = new List<Business>();
    test.Add(new Hotel { Name = "Hilton", Stars = 5 });
    test.Add(new Pool { Name = "Big Splash", Capacity = 500 });

    test.Dump();

    string json = JsonConvert.SerializeObject(test, Newtonsoft.Json.Formatting.Indented, new JsonSerializerSettings
    {
        TypeNameHandling = TypeNameHandling.All
    });

    json.Dump();

    var businesses = JsonConvert.DeserializeObject<List<Business>>(json, new JsonSerializerSettings
    {
        TypeNameHandling = TypeNameHandling.All
    });

    businesses.Dump();
}

// Define other methods and classes here
public abstract class Business
{
    public string Name { get;set; }
}
public class Hotel : Business
{
    public int Stars { get;set; }
}
public class Pool : Business
{
    public int Capacity { get;set;}
}

效果很好。抽象集合序列化为:

{
  "$type": "System.Collections.Generic.List`1[[UserQuery+Business, query_jvrdcu]], mscorlib",
  "$values": [
    {
      "$type": "UserQuery+Hotel, query_jvrdcu",
      "Stars": 5,
      "Name": "Hilton"
    },
    {
      "$type": "UserQuery+Pool, query_jvrdcu",
      "Capacity": 500,
      "Name": "Big Splash"
    }
  ]
}

原始集合和反序列化集合匹配。

【讨论】:

  • 其实是这样的,但是抽象类中还有一个list,因为它是一个树节点结构。
  • @L.Trabacchin 我已经发布了工作示例代码(如果您在“C# 程序”模式下使用 LINQPad 并引用 Newtonsoft.Json)。
  • 如果你编辑它以使用装饰器而不是设置我会接受你的回答
猜你喜欢
  • 2014-05-07
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多