NewtonSoft.Json Сериализация и десериализация класса со свойством типа IEnumerable

Я пытаюсь переместить некоторый код для использования веб-API ASP.NET MVC, сгенерированного данными Json, вместо SOAP Xml.

У меня возникла проблема с сериализацией и десериализацией свойств типа:

IEnumerable<ISomeInterface>.

Вот простой пример:

public interface ISample{
  int SampleId { get; set; }
}
public class Sample : ISample{
  public int SampleId { get; set; }
}
public class SampleGroup{
  public int GroupId { get; set; }
  public IEnumerable<ISample> Samples { get; set; }
 }
}

Я могу легко сериализовать экземпляры SampleGroup с помощью:

var sz = JsonConvert.SerializeObject( sampleGroupInstance );

Однако соответствующая десериализация завершается ошибкой:

JsonConvert.DeserializeObject<SampleGroup>( sz );

с этим сообщением об исключении:

«Не удалось создать экземпляр типа JsonSerializationExample.ISample. Тип — это интерфейс или абстрактный класс, и его нельзя создать».

Если я получаю JsonConverter, я могу украсить свое свойство следующим образом:

[JsonConverter( typeof (SamplesJsonConverter) )]
public IEnumerable<ISample> Samples { get; set; }

Вот JsonConverter:

public class SamplesJsonConverter : JsonConverter{
  public override bool CanConvert( Type objectType ){
    return ( objectType == typeof (IEnumerable<ISample>) );
  }

  public override object ReadJson( JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer ){
    var jA = JArray.Load( reader );
    return jA.Select( jl => serializer.Deserialize<Sample>( new JTokenReader( jl ) ) ).Cast<ISample>( ).ToList( );
  }

  public override void WriteJson( JsonWriter writer, object value, JsonSerializer serializer ){
   ... What works here?
  }
}

Этот преобразователь решает проблему десериализации, но я не могу понять, как закодировать метод WriteJson, чтобы сериализация снова заработала.

Кто-нибудь может помочь?

Является ли это «правильным» способом решения проблемы в первую очередь?

63
задан James Newton-King 13 November 2012 в 21:38
поделиться