Сериализация типов значений с помощью json.net

Я пытаюсь сделать вызов ajax, чтобы получить штаты из страны, но я продолжаю получать исключение invalidcastException. Я использую MVC4 с json.NET. Я могу без проблем сериализовать объекты в своих тестах, но когда я делаю вызов ajax, я продолжаю получать сообщение об ошибке.

Это тип значения:

[JsonConverter(typeof(ValueObjectConverter))]
public class Code : IValueObject
{
    private readonly string _code;

    private Code(string code)
    {
        _code = code;
    }

    public override string ToString()
    {
        return _code;
    }

    public static implicit operator Code(string code)
    {
        return new Code(code);
    }

    public static implicit operator String(Code code)
    {
        return code.ToString();
    }
}

Это мой объект:

public class CountryState : Entity
{
    public Code CodeWPS { get; set; }
    public Code CodeIAM { get; set; }
    public Name Description { get; set; }
}

Это jsonconverter:

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

    public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
    {
        return (IValueObject)existingValue;
    }

    public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
    {
        var item = (IValueObject)value;
        writer.WriteValue(item.ToString());
        writer.Flush();
    }
}

Это метод API:

    public IEnumerable<CountryState> GetStatesByCountry(string codeType, string countryCodeIam)
    {
        var states = _getAllCountriesQueryHandler.Handle(countryCodeIam);
        return states;
    }

Это ошибка, которую я получаю с firebug:

    {"Message":"An error has occurred.","ExceptionMessage":"The 'ObjectContent`1' type    failed to serialize the response body for content type 'application/json; charset=utf-8'.","ExceptionType":"System.InvalidOperationException","StackTrace":null,"InnerException":{"Message":"An error has occurred.","ExceptionMessage":"Error getting value from 'CodeWPS' on 'CNH.CSCN.BBS.Entities.CountryState'.","ExceptionType":"Newtonsoft.Json.JsonSerializationException","StackTrace":"   at Newtonsoft.Json.Serialization.DynamicValueProvider.GetValue(Object target)\r\n   at Newtonsoft.Json.Serialization.JsonSerializerInternalWriter.CalculatePropertyValues(JsonWriter writer, Object value, JsonContainerContract contract, JsonProperty member, JsonProperty property, JsonContract& memberContract, Object& memberValue)\r\n   at Newtonsoft.Json.Serialization.JsonSerializerInternalWriter.SerializeObject(JsonWriter writer, Object value, JsonObjectContract contract, JsonProperty member, JsonContainerContract collectionContract, JsonProperty containerProperty)\r\n   at Newtonsoft.Json.Serialization.JsonSerializerInternalWriter.SerializeList(JsonWriter writer, IEnumerable values, JsonArrayContract contract, JsonProperty member, JsonContainerContract collectionContract, JsonProperty containerProperty)\r\n   at Newtonsoft.Json.Serialization.JsonSerializerInternalWriter.Serialize(JsonWriter jsonWriter, Object value, Type objectType)\r\n   at Newtonsoft.Json.JsonSerializer.SerializeInternal(JsonWriter jsonWriter, Object value, Type objectType)\r\n   at System.Net.Http.Formatting.JsonMediaTypeFormatter.<>c__DisplayClassd.<WriteToStreamAsync>b__c()\r\n   at System.Threading.Tasks.TaskHelpers.RunSynchronously(Action action, CancellationToken token)","InnerException":{"Message":"An error has occurred.","ExceptionMessage":"Unable to cast object of type 'System.String' to type 'CNH.CSCN.BBS.Entities.ValueTypes.Code'.","ExceptionType":"System.InvalidCastException","StackTrace":"   at GetCodeWPS(Object )\r\n   at Newtonsoft.Json.Serialization.DynamicValueProvider.GetValue(Object target)"}}}

ОБНОВИТЬ:

Я добавил контрактный преобразователь:

public class SpecialContractResolver : DefaultContractResolver
{
    protected override IValueProvider CreateMemberValueProvider(MemberInfo member)
    {
        if (member.MemberType == MemberTypes.Property)
        {
            var pi = (PropertyInfo)member;
            if (typeof(IValueObject).IsAssignableFrom(pi.PropertyType))
            {
                return new IValueObjectValueProvider(member, pi.PropertyType);
            }
        }
        else if (member.MemberType == MemberTypes.Field)
        {
            var fi = (FieldInfo)member;
            if (typeof(IValueObject).IsAssignableFrom(fi.FieldType))
            {
                return new IValueObjectValueProvider(member, fi.FieldType);
            }
        }

        return base.CreateMemberValueProvider(member);
    }
}

и IValueProvider:

public class IValueObjectValueProvider : IValueProvider
{
    private readonly object _defaultValue;
    private readonly IValueProvider _underlyingValueProvider;


    public IValueObjectValueProvider(MemberInfo memberInfo, Type underlyingType)
    {
        _underlyingValueProvider = new DynamicValueProvider(memberInfo);
        _defaultValue = underlyingType;
    }

    public void SetValue(object target, object value)
    {
        target = value;
    }

    public object GetValue(object target)
    {
        return target.ToString();
    }
}

Я знаю, что я на правильном пути, я больше не получаю сообщение об ошибке, но такой результат:

Объект { CodeWPS="CNH.CSCN.BBS.Entities.CountryState", CodeIAM="CNH.CSCN.BBS.Entities.CountryState", Description="CNH.CSCN.BBS.Entities.CountryState"}

Я думаю, что мой contractresolver и ivalueprovider еще не на 100% верны...


person Selketjah    schedule 09.08.2013    source источник
comment
После дальнейшего расследования выяснилось, что он не использует мой собственный конвертер при вызове ajax.   -  person Selketjah    schedule 09.08.2013


Ответы (2)


Это из-за этого фрагмента кода:

public override bool CanConvert(Type objectType)
{
    return (objectType == typeof(IValueObject));
}

Ваш objectType не IValueObject, это Code, поэтому CanConvert возвращает false и ваш преобразователь не используется. Ваш код должен выглядеть так:

return typeof(IValueObject).IsAssignableFrom(objectType);

С этим условием также сопоставляются реализующие классы.

person fero    schedule 12.08.2013
comment
Спасибо за подсказку, к сожалению, это не устраняет мое недопустимое исключение приведения. CanConvert получает следующие входные данные: список типов - тип страны - тип httperror - тип строки... - person Selketjah; 12.08.2013
comment
Вы получаете ошибку приведения, потому что пытаетесь распаковать existingValue в IValueObject, что невозможно, потому что оно еще не десериализовано. Вы должны сначала десериализовать JSON в строку C#, а затем преобразовать его в объект Code, что довольно сложно, поскольку у вас есть только оператор неявного приведения типа для создания нового объекта Code. Если ваш конструктор был общедоступным, вы можете написать что-то вроде return Activator.CreateInstance(objectType, serializer.Deserialize<string>(reader)); или использовать другую перегрузку CreateInstance для использования закрытого конструктора. - person fero; 12.08.2013
comment
Он никогда не достигает моих методов ReadJson или WriteJson, я прямо сейчас изучаю DynamicValueProvider JSON.NET, чтобы увидеть, как это работает... - person Selketjah; 13.08.2013

Я создал SpecialContractResolver

public class SpecialContractResolver : DefaultContractResolver
{
    protected override IValueProvider CreateMemberValueProvider(MemberInfo member)
    {
        if (member.MemberType == MemberTypes.Property)
        {
            var pi = (PropertyInfo)member;
            if (typeof(IValueObject).IsAssignableFrom(pi.PropertyType))
            {
                return new ExpressionValueProvider(member);
            }
        }

        return base.CreateMemberValueProvider(member);
    }
}

Я создал ValueObjectConverter:

public class ValueObjectConverter : JsonConverter
{
    public override bool CanConvert(Type objectType)
    {
        return typeof(IValueObject).IsAssignableFrom(objectType);
    }

    public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
    {
        return Activator.CreateInstance(objectType, serializer.Deserialize<string>(reader));
    }

    public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
    {
        var item = (IValueObject)value;
        writer.WriteValue(item.ToString());
        writer.Flush();
    }
}

И я добавил его в свой WebApiConfig:

        var jsonFormatter = config.Formatters.JsonFormatter;
        jsonFormatter.SerializerSettings.ContractResolver = new SpecialContractResolver();
        jsonFormatter.SerializerSettings.Converters.Insert(0, new ValueObjectConverter());

Надеюсь, это поможет кому-то!

person Selketjah    schedule 13.08.2013