反序列化泛型类型



我有一个泛型类型Container<IContentType>其中接口IContentType可以是四种具体的ContentX类型之一。

我序列化,一切都很好。

使用 Newtonsoft 反序列化时,我使用自定义类型转换器并且var model = JsonConvert.DeserializeObject<Container<ContentA>>(json, settings)工作。调试器显示我有一个Container<ContentA>对象。

我的计划是在反序列化时尝试对四种可能的ContentX类型中的每一种进行反序列化,并静默地捕获异常,直到我"猜出"正确的异常。

但是,如果我在这样的方法中执行此操作:

public static Container<IContentType> Deserialize(jsonfile)
{
    ...
    var model = JsonConvert.DeserializeObject<Container<ContentA>>(json, settings)
    return model;
}

我得到"无法隐式将Container<ContentA>转换为Container<IContentType>"。 ContentA实现IContentType .

有没有办法创建强制转换运算符、转换、动态或使隐式转换工作?

与其

尝试反序列化为具体类型XContainer<ContentX> Container<IContentType>,不如使用自定义JsonConverter将 JSON 预加载到JToken中并按照 如何在 JSON.NET 中实现自定义 JsonConverter 以反序列化基类对象列表? 或使用 json.net 或 Json.Net 反序列化没有类型信息的多态 JSON 类 使用多态类型序列化子对象。

因此,您的转换器将如下所示:

public class ContentConverter : JsonConverter
{
    public override bool CanConvert(Type objectType)
    {
        return objectType == typeof(IContentType);
    }
    Type GetConcreteType(JObject obj)
    {
        if (obj.GetValue(nameof(ContentA.SomePropertyOfContentA), StringComparison.OrdinalIgnoreCase) != null)
            return typeof(ContentA);
        // Add other tests for other content types.
        // Return a default type or throw an exception if a unique type cannot be found.
        throw new JsonSerializationException("Cannot determine concrete type for IContentType");
    }
    public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
    {
        if (reader.TokenType == JsonToken.Null)
            return null;
        var obj = JObject.Load(reader);
        var concreteType = GetConcreteType(obj);
        return obj.ToObject(concreteType, serializer);
    }
    public override bool CanWrite { get { return false; } }
    public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
    {
        throw new NotImplementedException();
    }
}

你对JsonConvert的调用看起来像:

var settings = new JsonSerializerSettings
{
    Converters = { new ContentConverter() },
};
var model = JsonConvert.DeserializeObject<Container<IContentType>>(json, settings);

最后,您可以使用

new JsonDerivedTypeConverer<IContentType>(typeof(ContentA), typeof(ContentB), typeof(ContentC), typeof(ContentD))

其中JsonDerivedTypeConverer<T>取自 JsonConverter with Interface。

相关内容

  • 没有找到相关文章

最新更新