Json.net反序列化列表提供重复项

时间:2012-11-15 09:15:36

标签: c# json json.net

我刚开始使用Newtonsoft.Json(Json.net)。在我的第一个简单测试中,我在反序列化通用列表时遇到了问题。在下面的代码示例中,我序列化了一个对象,包含三种类型的简单整数列表(属性,成员var和数组)。

生成的json看起来很好(列表转换为json数组)。但是,当我将json反序列化为相同类型的新对象时,所有列表项都是重复的,期望数组。我已经通过第二次序列化来说明了这一点。

从搜索中,我已经读到反序列化器也会填充的列表中可能存在“私有”支持字段。

所以我的问题是:在下列情况下是否有(最好是简单的)方法来避免重复项目?

代码

using System;
using System.Collections.Generic;
using Newtonsoft.Json;

namespace JsonSerializeExample
{
    public class Program
    {
        static void Main()
        {
            var data = new SomeData();
            var json = JsonConvert.SerializeObject(data);
            Console.WriteLine("First : {0}", json);
            var data2 = JsonConvert.DeserializeObject<SomeData>(json);
            var json2 = JsonConvert.SerializeObject(data2);
            Console.WriteLine("Second: {0}", json2);
        }
    }

    public class SomeData
    {
        public string SimpleField;
        public int[] IntArray;
        public IList<int> IntListProperty { get; set; }
        public IList<int> IntListMember;

        public SomeData()
        {
            SimpleField = "Some data";
            IntArray = new[] { 7, 8, 9 };
            IntListProperty = new List<int> { 1, 2, 3 };
            IntListMember = new List<int> { 4, 5, 6 };
        }
    }
}

产生的输出

First : {"SimpleField":"Some data","IntArray":[7,8,9],"IntListMember":[4,5,6],"IntListProperty":[1,2,3]}
Second: {"SimpleField":"Some data","IntArray":[7,8,9],"IntListMember":[4,5,6,4,5,6],"IntListProperty":[1,2,3,1,2,3]}

此处可能与Json.Net duplicates private list items存在一些重叠。但是,我认为我的问题更简单,我仍然没有想到它。

4 个答案:

答案 0 :(得分:21)

这是因为您在构造函数中添加项目。处理列表时反序列化器中的常用方法基本上是:

  • 通过getter读取列表
    • 如果列表为null:创建一个新列表并通过属性setter分配,如果是
  • 依次反序列化每个项目,并将(Add)追加到列表

这是因为大多数列表成员没有setter ,即

public List<Foo> Items {get {...}} // <=== no set

与数组形成对比,数组必须有一个有用的setter;因此,方法通常是:

  • 依次反序列化每个项目,并将(Add)附加到临时列表
  • 将列表转换为数组(ToArray),并通过setter
  • 进行分配

某些序列化程序为您提供了控制此行为的选项(其他人没有);并且一些序列化程序使您能够完全绕过构造函数(其他人没有)。

答案 1 :(得分:7)

我遇到了一个不同根本原因的类似问题。我正在序列化和反序列化一个看起来像这样的类:

public class Appointment
{
    public List<AppointmentRevision> Revisions { get; set; }

    public AppointmentRevision CurrentRevision
    {
        get { return Revision.LastOrDefault(); }
    }

    public Appointment()
    {
        Revisions = new List<AppointmentRevision>();
    }
}

public class AppointmentRevision
{
    public List<Attendee> Attendees { get; set; }
}

当我序列化时,CurrentRevision也被序列化了。我不确定如何,但是当它反序列化时,它正确地保留了AppointmentRevision的单个实例,但在Attendees列表中创建了重复项。解决方案是在CurrentRevision属性上使用JsonIgnore属性。

public class Appointment
{
    public List<AppointmentRevision> Revisions { get; set; }

    [JsonIgnore]   
    public AppointmentRevision CurrentRevision
    {
        get { return Revision.LastOrDefault(); }
    }

    public Appointment()
    {
        Revisions = new List<AppointmentRevision>();
    }
}

答案 2 :(得分:3)

我很确定这篇文章不再相关,但是为了将来参考,这里是一个可行的解决方案。 只需指定将--safe设置为ObjectCreationHandling,即始终创建新对象而不是Replace(这是默认设置),即重用现有对象,在需要时创建新对象。 / p>

Auto

答案 3 :(得分:0)

How to apply ObjectCreationHandling.Replace to selected properties when deserializing JSON?

结果证明(我在2019年),您可以像在问题中一样在构造函数中设置列表项。我在列表声明的上方添加了ObjectCreationHandling.Replace属性,然后序列化应使用JSON替换列表中存储的任何内容。