Функция JsonConvert.DeserializeObject выдает ошибки при разборе устаревших свойств перечисления

Я хочу пометить существующее свойство Enum как устаревшее и добавить новое для той же функциональности. Итак, два перечисления (новое и устаревшее помечено одно) доступно в моем проекте.

При использовании функции jsonconvert.deserializeobject для сериализации это значение перечисления выдает следующие ошибки только из вышеприведенной версии Newtonsoft 11.0.

InvalidOperationException: Enum name 'stringedit' already exists on enum 'EditingType'. Newtonsoft.Json.Utilities.EnumUtils.InitializeValuesAndNames(Type enumType) JsonSerializationException: Error converting value "boolean" to type 'Student.Models.EditingType'. Path 'columns[0].editType', line 1, position 3997. Newtonsoft.Json.Serialization.JsonSerializerInternalReader.EnsureType(JsonReader reader, object value, CultureInfo culture, JsonContract contract, Type targetType)

Если есть какие-то изменения, которые нужно сделать в моем проекте, или если есть какой-либо план по устранению этой новой ошибки из функции сериализации.

Я использовал приведенный ниже блок кода для своей устаревшей сериализации свойства enum.

[DataContract]

public enum EditingType

{

    [Obsolete("String property has been deprecated. Use StringEdit property instead")]

    [EnumMember(Value = "stringedit")]

    String,

    [EnumMember(Value = "stringedit")]

    StringEdit

}

public class JsonPropertyAttribute : Attribute

{
    public JsonPropertyAttribute(string argName)
    {
        PropertyName = argName;
    }

    public string PropertyName { get; set; }
}

public class JsonConverterAttribute : Attribute
{
    public Type ConverterType
    {
        get; set;
    }
    public string ValueType
    {
        get; set;
    }
    public JsonConverterAttribute(Type type, string valueType)
    {
        this.ConverterType = type;
        this.ValueType = valueType;
    }
    public JsonConverterAttribute(Type type)
    {
        this.ConverterType = type;

    }

}

public abstract class Converter
{
    protected internal abstract IDictionary<string, object> BuildJsonDictionary(object value);

    public abstract string SerializeToJson(object inputObject);
}
public class StringEnumConverter : Converter
{
    protected internal override IDictionary<string, object> BuildJsonDictionary(object value)
    {

        IDictionary<string, object> jsonDictionary = new Dictionary<string, object>();
        Type objectType = value.GetType();

        FieldInfo member = objectType.GetField(value.ToString(), BindingFlags.Static | BindingFlags.Public);

        var attrList = member.GetCustomAttributes(typeof(EnumMemberAttribute), true);
        var enumMember = attrList.OfType<EnumMemberAttribute>().FirstOrDefault();

        string val = enumMember != null ? enumMember.Value : value.ToString();

        jsonDictionary.Add(value.GetType().Name, val);

        return jsonDictionary;
    }


    public override string SerializeToJson(object inputObject)
    {
        var attrList = inputObject.GetType().GetCustomAttributes(false);
        var listAttr = attrList.ToList();
        FlagsAttribute flagAttr = attrList.Count() != 0 ? (FlagsAttribute)listAttr.Find(item => item.GetType() == typeof(FlagsAttribute)) : null;
        bool flag = (flagAttr != null) ? true : false;
        if (flag)
        {
            int value = (int)inputObject;
            return value.ToString();
        }
        else
        {
            IDictionary<string, object> enumDictionary = BuildJsonDictionary(inputObject);
            object enumValue = enumDictionary.First().Value;
            string enumstring = "\"" + enumValue.ToString() + "\"";
            return enumstring;
        }
    }
}

[Serializable]
public class TestClass
{
    private EditingType _editingType = EditingType.StringEdit;

    [JsonProperty("editType")]
    [DefaultValue(EditingType.StringEdit)]
    [JsonConverter(typeof(StringEnumConverter))]
    public EditingType EditType
    {
        get { return _editingType; }
        set { _editingType = value; }
    }
}

Приведенная ниже функция десериализации выдаст ошибку в последнем NewtonSoft.json (начиная с версии 11.X)

public class HomeController : Controller
{
    public IActionResult Index()
    {
       // This function will throws the error
        JsonConvert.DeserializeObject("{\"editType\":\"string\"}", typeof(TestClass));
        return View();
    }
}

Спасибо..

1 ответ

Я посмотрел на это, и это ожидаемое поведение. Json.NET теперь всегда заботится о EnumMember, и у вас есть повторяющиеся значения.

Реализуйте JsonConverter, если вы хотите настроить его сериализацию. Я не знаю, что вы пытаетесь сделать с классами преобразования в вашем образце, но Json.NET их не волнует.

Цитировать автора Json.NET по ссылке @dbc на GitHub

Другие вопросы по тегам