Json.Net中的PreserveReferencesHandling和ReferenceLoopHandling有什么区别?

Json.Net中的PreserveReferencesHandling和ReferenceLoopHandling有什么区别?,第1张

Json.Net中的PreserveReferencesHandling和ReferenceLoopHandling有什么区别?

这些设置可以通过示例来最好地解释。假设我们要表示公司中员工的层次结构。因此,我们创建了一个简单的类,如下所示:

class Employee{    public string Name { get; set; }    public List<Employee> Subordinates { get; set; }}

这是一家只有三名员工的小公司:Angela,Bob和Charles。安吉拉是老板,而鲍勃和查尔斯是她的下属。让我们设置数据来描述这种关系:

Employee angela = new Employee { Name = "Angela Anderson" };Employee bob = new Employee { Name = "Bob Brown" };Employee charles = new Employee { Name = "Charles Cooper" };angela.Subordinates = new List<Employee> { bob, charles };List<Employee> employees = new List<Employee> { angela, bob, charles };

如果我们将员工列表序列化为JSON …

string json = JsonConvert.SerializeObject(employees, Formatting.Indented);Console.WriteLine(json);

…我们得到以下输出:

[  {    "Name": "Angela Anderson",    "Subordinates": [      {        "Name": "Bob Brown",        "Subordinates": null      },      {        "Name": "Charles Cooper",        "Subordinates": null      }    ]  },  {    "Name": "Bob Brown",    "Subordinates": null  },  {    "Name": "Charles Cooper",    "Subordinates": null  }]

到目前为止,一切都很好。但是,您会注意到,在JSON中重复了Bob和Charles的信息,因为代表员工的对象同时被雇员的主要列表和Angela的下属列表引用。也许现在还可以。

现在,假设我们还希望有一种方法来跟踪每个员工的主管以及下属。因此,我们更改了

Employee
模型以添加
Supervisor
属性…

class Employee{    public string Name { get; set; }    public Employee Supervisor { get; set; }    public List<Employee> Subordinates { get; set; }}

…并在我们的设置代码中再添加几行,以表明Charles和Bob向Angela报告:

Employee angela = new Employee { Name = "Angela Anderson" };Employee bob = new Employee { Name = "Bob Brown" };Employee charles = new Employee { Name = "Charles Cooper" };angela.Subordinates = new List<Employee> { bob, charles };bob.Supervisor = angela;       // added this linecharles.Supervisor = angela;   // added this lineList<Employee> employees = new List<Employee> { angela, bob, charles };

但是现在我们有点问题了。因为对象图中有引用循环(例如,

angela
引用
bob
时为
bob
reference
angela
),所以
JsonSerializationException
当我们尝试序列化雇员列表时,我们将得到一个。解决此问题的一种方法是设置
ReferenceLoopHandling
Ignore

JsonSerializerSettings settings = new JsonSerializerSettings{    ReferenceLoopHandling = ReferenceLoopHandling.Ignore,    Formatting = Formatting.Indented};string json = JsonConvert.SerializeObject(employees, settings);

完成此设置后,我们将获得以下JSON:

[  {    "Name": "Angela Anderson",    "Supervisor": null,    "Subordinates": [      {        "Name": "Bob Brown",        "Subordinates": null      },      {        "Name": "Charles Cooper",        "Subordinates": null      }    ]  },  {    "Name": "Bob Brown",    "Supervisor": {      "Name": "Angela Anderson",      "Supervisor": null,      "Subordinates": [        {          "Name": "Charles Cooper",          "Subordinates": null        }      ]    },    "Subordinates": null  },  {    "Name": "Charles Cooper",    "Supervisor": {      "Name": "Angela Anderson",      "Supervisor": null,      "Subordinates": [        {          "Name": "Bob Brown",          "Subordinates": null        }      ]    },    "Subordinates": null  }]

如果您检查JSON,则应该清楚此设置的作用:只要序列化程序遇到对对象的引用,该对象已经在序列化过程中,它只会跳过该成员。(这防止了序列化程序陷入无限循环。)您可以看到,在JSON顶部的Angela的下属列表中,Bob和Charles都不显示主管。在JSON的底部,Bob和Charles都显示了Angela为他们的主管,但是请注意,此时她的下属列表并不同时包括Bob和Charles。

虽然可以使用此JSON,甚至可以通过一些工作从中重建原始对象层次结构,但这显然不是最佳的。我们可以通过使用

PreserveReferencesHandling
设置来消除JSON中的重复信息,同时仍然保留对象引用:

JsonSerializerSettings settings = new JsonSerializerSettings{    PreserveReferencesHandling = PreserveReferencesHandling.Objects,    Formatting = Formatting.Indented};string json = JsonConvert.SerializeObject(employees, settings);

现在我们得到以下JSON:

[  {    "$id": "1",    "Name": "Angela Anderson",    "Supervisor": null,    "Subordinates": [      {        "$id": "2",        "Name": "Bob Brown",        "Supervisor": {          "$ref": "1"        },        "Subordinates": null      },      {        "$id": "3",        "Name": "Charles Cooper",        "Supervisor": {          "$ref": "1"        },        "Subordinates": null      }    ]  },  {    "$ref": "2"  },  {    "$ref": "3"  }]

请注意,现在已为每个对象

$id
在JSON中分配了一个顺序值。第一次出现某个对象时,它会被完全序列化,而随后的引用将被一个特殊
$ref
属性替换,该属性会使用相应的属性返回原始对象
$id
。有了这个设置,JSON是更简洁,可以不需要额外的工作进行反序列化回原来的对象的层次结构,假设你使用的是理解图书馆
$id
$ref
通过Json.Net
/网络API产生的符号。

那么,为什么要选择一种设置呢?当然,这取决于您的需求。如果JSON将由不了解

$id
/
$ref
格式的客户端使用,并且可以容忍某些地方存在不完整的数据,则可以选择使用
ReferenceLoopHandling.Ignore
。如果您正在寻找更紧凑的JSON,并且将使用Json.Net或Web
API(或另一个兼容的库)对数据进行反序列化,则可以选择使用
PreserveReferencesHandling.Objects
。如果您的数据是没有重复引用的有向无环图,则无需任何设置。



欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/zaji/5018436.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022-11-14
下一篇 2022-11-15

发表评论

登录后才能评论

评论列表(0条)

保存