加入收藏 | 设为首页 | 会员中心 | 我要投稿 李大同 (https://www.lidatong.com.cn/)- 科技、建站、经验、云计算、5G、大数据,站长网!
当前位置: 首页 > 百科 > 正文

c# – 使用JsonConverterAttribute时,自定义继承JsonConverter失

发布时间:2020-12-16 02:03:00 所属栏目:百科 来源:网络整理
导读:我试图反序列化派生类型,我想使用自定义属性Type来区分派生类型. [ { "Type": "a","Height": 100 },{ "Type": "b","Name": "Joe" }] 我遇到的解决方案是创建一个自定义的JsonConverter.在ReadJson上,我读取了Type属性,并通过ToObject T实例化该类型.功能.一
我试图反序列化派生类型,我想使用自定义属性Type来区分派生类型.

[
  {
    "Type": "a","Height": 100
  },{
    "Type": "b","Name": "Joe"
  }
]

我遇到的解决方案是创建一个自定义的JsonConverter.在ReadJson上,我读取了Type属性,并通过ToObject< T>实例化该类型.功能.一切正常,直到我使用JsonConverterAttribute. ReadJson方法无限循环,因为该属性也应用于子类型.

如何防止将此属性应用于子类型?

[JsonConverter(typeof(TypeSerializer))]
public abstract class Base
{
    private readonly string type;

    public Base(string type)
    {
        this.type = type;
    }

    public string Type { get { return type; } }
}

public class AType : Base
{
    private readonly int height;

    public AType(int height)
        : base("a")
    {
        this.height = height;
    }

    public int Height { get { return height; } }
}

public class BType : Base
{
    private readonly string name;

    public BType(string name)
        : base("b")
    {
        this.name = name;
    }

    public string Name { get { return name; } }
}

public class TypeSerializer : JsonConverter
{
    public override bool CanConvert(Type objectType)
    {
        return objectType == typeof(Base);
    }

    public override void WriteJson(JsonWriter writer,object value,JsonSerializer serializer)
    {
        serializer.Serialize(writer,value);
    }

    public override object ReadJson(JsonReader reader,Type objectType,object existingValue,JsonSerializer serializer)
    {
        var j = JObject.Load(reader);

        var type = j["Type"].ToObject<string>();

        if (type == "a")
            // Infinite Loop! StackOverflowException
            return j.ToObject<AType>(); 
        if (type == "b")
            return j.ToObject<BType>();

        throw new NotImplementedException(type);
    }
}

[TestFixture]
public class InheritanceSerializeTests
{
    [Test]
    public void Deserialize()
    {
        var json = @"{""Type"":""a"",""Height"":100}";
        JObject.Parse(json).ToObject<Base>(); // Crash
    }
}

解决方法

我当前正在研究的项目有一个非常类似的问题:我想制作一个自定义的JsonConverter并通过属性将它映射到我的实体,但随后代码被困在一个无限循环中.

在我的情况下,诀窍是使用serializer.Populate而不是JObject.ToObject(即使我想要,也无法使用.ToObject;我使用的是版本3.5.8,其中此函数不存在).下面以我的ReadJson方法为例:

public override object ReadJson(JsonReader reader,JsonSerializer serializer)
{
    JContainer lJContainer = default(JContainer);

    if (reader.TokenType == JsonToken.StartObject)
    {
        lJContainer = JObject.Load(reader);
        existingValue = Convert.ChangeType(existingValue,objectType);
        existingValue = Activator.CreateInstance(objectType);

        serializer.Populate(lJContainer.CreateReader(),existingValue);
    }

    return existingValue;
}

(编辑:李大同)

【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!

    推荐文章
      热点阅读