带有嵌套对象的JSON.NET CustomCreationConverter

CGR*_*CGR 5 c# serialization json.net object-graph json-deserialization

这是我在这个网站上提出的第一个问题,如果我错过了什么,请原谅我.

我在使用JSON.NET反序列化复杂对象图时遇到了一些问题.我的类层次结构(简化)如下:

public abstract class BusinessObjectBase
{
    protected BusinessObjectBase(SerializationContext context)
    {
    }
}

public class TestBusinessObject : BusinessObjectBase
{
    protected TestBusinessObject(SerializationContext context)
        : base(context)
    {
    }

    public NestedObject InnerObject { get; set; }
}

public class NestedObject : BusinessObjectBase
{
    protected NestedObject(SerializationContext context)
        : base(context)
    {
    }
}
Run Code Online (Sandbox Code Playgroud)

这些类没有默认的ctor,而是一个专用的自定义反序列化ctor(除了带参数的其他公共ctor),如示例所示.要创建一个实例,我编写了一个自定义创建转换器,如下所示:

internal class BusinessObjectCreationConverter : CustomCreationConverter<BusinessObjectBase>
{
    public override bool CanConvert(Type objectType)
    {
        return typeof(BusinessObjectBase).IsAssignableFrom(objectType) && !objectType.IsAbstract;
    }

    public override BusinessObjectBase Create(Type objectType)
    {
        var businessObject = objectType.CreateUsingDesrializationConstructor<BusinessObjectBase>();
        return businessObject;
    }
}
Run Code Online (Sandbox Code Playgroud)

CreateUsingDesrializationConstructor()扩展方法查找特殊反序列化ctor并使用ctor创建实例.

我将转换器添加到我的JSON.NET序列化程序实例:

public class NewtonsoftJsonSerializer : ISerializer
{
    public NewtonsoftJsonSerializer()
        : this(new JsonSerializer
        {
            TypeNameHandling = TypeNameHandling.Auto,
            ObjectCreationHandling = ObjectCreationHandling.Replace,
            PreserveReferencesHandling = PreserveReferencesHandling.Objects,
            ConstructorHandling = ConstructorHandling.AllowNonPublicDefaultConstructor,
            DefaultValueHandling = DefaultValueHandling.Ignore,
            ContractResolver = new KsJsonContractResolver()
        })
    {
        this.serializer.Converters.Add(new BusinessObjectCreationConverter());
    }

    public T Deserialize<T>(Stream stream)
    {
        T result;
        using (var streamReader = new StreamReader(stream, Encoding.UTF8, true, BufferSize, true))
        using (var jsonReader = new JsonTextReader(streamReader))
        {
            result = this.serializer.Deserialize<T>(jsonReader);
        }

        return result;
    }
}
Run Code Online (Sandbox Code Playgroud)

当我反序列化TestBusinessObject时,我可以从调试器看到转换器询问每种类型是否能够创建实例:TestBusinessObject,NestedObject和许多其他类型.但我的转换器只是被要求创建一个新的TestBusinessObject实例,他没有被要求创建我所期望的嵌套NestedObject实例和我非常需要的,因为反序列化ctor中有一些有线逻辑.

我在这里做错了,如何告诉JsonSerializer将转换器用于每个对象,甚至不是根(顶层)对象?

编辑: 当一个BusinessObjectBase实例包含在我不知道的类型的对象中时,认为变得更加复杂.在这种情况下,我也希望调用转换器.

提前谢谢,卡斯滕

小智 0

尝试为要序列化或反序列化的类提供 [DataContract] 属性,并确保这些类中拥有的任何数据也具有此属性。例如

[DataContract]
public class TestBusinessObject : BusinessObjectBase
{
    protected TestBusinessObject(SerializationContext context)
        : base(context)
    {{
    }

    public NestedObject InnerObject { get; set; }
}
Run Code Online (Sandbox Code Playgroud)