public class SpecialObject
{
public string ID;
[JsonIgnore]
public List<SpecialObject> SpecialObjectCollection = new List<SpecialObject>();
[JsonIgnore]
public List<string> tempObjectIDs = new List<string>();
[JsonProperty]
public List<string> SpecialObjectIDs { get { return SpecialObjectCollection.Select(x => x.ID).ToList(); } set { tempObjectIDs = value; } }
public SpecialObject() { }
public SpecialObject(string _id) { ID = _id; }
}
static void Main(string[] args)
{
SpecialObject parent = new SpecialObject("parentIDstring");
parent.SpecialObjectCollection.Add(new SpecialObject("childIDstring"));
string test = JsonConvert.SerializeObject(parent);
SpecialObject reconstructedObject = JsonConvert.DeserializeObject<SpecialObject>(test);
}
//string test:
//{"ID":"parentIDstring","SpecialObjectIDs":["childIDstring"]}
我想将SpecialObject序列化为JSON并再次返回。 &#34; SpecialObjectIDs&#34;访问器让我避免讨厌的循环引用,并帮助我重建复杂的关系网络,因为我只需要唯一的ID。
它序列化很好,但我需要反序列化对象;当我这样做时,我的数组(JSON中的SpecialObjectIDs)在转换过程中消失,似乎没有调用set访问器。
如何让Newtonsoft.Json(JSON.net)库将反序列化的List放在tempObjectIDs中,但保留现有的get行为?
答案 0 :(得分:2)
您的问题是,在反序列化非只读集合时,Json.NET会检查集合是否已经分配,例如在包含类型的构造函数中。如果是这样,它将填充反序列化的JSON内容的预先存在的集合,并且永远不会将集合设置回来。不幸的是,您的属性返回一个临时代理集合,因此您的容器类SpecialObject
永远不会收到反序列化的结果。
防止这种情况的最简单方法是指定Json.NET应始终分配新集合并将其设置回来,而不是通过JsonPropertyAttribute
设置ObjectCreationHandling = ObjectCreationHandling.Replace
重用预先存在的集合
[JsonProperty(ObjectCreationHandling = ObjectCreationHandling.Replace)]
public List<string> SpecialObjectIDs { get { return SpecialObjectCollection.Select(x => x.ID).ToList(); } set { tempObjectIDs = value; } }
或者,您可以使用string []
而不是List<string>
作为代理集合属性:
public string [] SpecialObjectIDs { get { return SpecialObjectCollection.Select(x => x.ID).ToArray(); } set { tempObjectIDs = value == null ? null : value.ToList(); } }
由于数组无法调整大小,Json.NET将在反序列化时始终分配一个新数组,并在完成时将其设置回来。