已申报的财产赢得';如果其声明类型是动态对象,则不能序列化

本文关键字:动态 类型 声明 对象 序列化 不能 如果 财产 | 更新日期: 2023-09-27 18:00:05

假设我有以下动态对象:

public class SomeDynamicObject : DynamicObject
{
     public string Text { get; set; }
}

如果我使用JsonConvert.SerializeObject(new SomeDynamicObject { Text = "hello world" })序列化它,它将返回{}而不是{ "Text": "hello world" }

我怀疑问题是JSON.NET认为它是一个完整的动态对象,而我的案例是一个具有声明成员的动态对象。

是否可以配置任何序列化设置或内置转换器,以便JSON.NET可以序列化这两种成员?

避免混淆

实际用例:我不知道哪些类型将被序列化,但我需要介绍序列化动态对象的声明属性的整个用例。

也就是说,我不能使用属性。这就是为什么我要问是否有一些转换器或序列化设置可以概括这个用例。

已申报的财产赢得';如果其声明类型是动态对象,则不能序列化

更新非属性转换器

既然你不能装修,你就会失去很多动力。一旦JsonWriter转换为JObject,动态属性就会丢失。

但是,您总是可以在自定义转换器的WriteJson方法中使用一点反射来序列化非动态类型。

public class SomeDynamicObject : DynamicObject
{
    public string Text { get; set; }
    public DynamicObject DynamicProperty { get; set; }
}
public class CustomDynamicConverter : JsonConverter
{
    public override bool CanConvert(Type objectType)
    {
        return true;
    }
    public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
    {
        if (reader.TokenType == JsonToken.Null)
            return null;
        JObject jObject = JObject.Load(reader);
        var target = Activator.CreateInstance(objectType);
        //Create a new reader for this jObject, and set all properties to match the original reader.
        JsonReader jObjectReader = jObject.CreateReader();
        jObjectReader.Culture = reader.Culture;
        jObjectReader.DateParseHandling = reader.DateParseHandling;
        jObjectReader.DateTimeZoneHandling = reader.DateTimeZoneHandling;
        jObjectReader.FloatParseHandling = reader.FloatParseHandling;
        // Populate the object properties
        serializer.Populate(jObjectReader, target);
        return target;
    }
    public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
    {
        var properties = value.GetType().GetProperties().Where(x => x.PropertyType != typeof(DynamicObject)).ToList();
        JObject o = (JObject)JToken.FromObject(value);
        properties.ForEach(x =>
        {
            o.AddFirst(new JProperty(x.Name, x.GetValue(value)));
        });
        o.WriteTo(writer);
    }
}

如果使用[JsonProperty]显式修饰属性,即使包含的类型是动态的,序列化程序也会拾取它们。

public class SomeDynamicObject : DynamicObject
{
    [JsonProperty]
    public string Text { get; set; }
}

当串行化正确输出时:

{"Text":"hello world"}