#c# #json #azure #serialization
#c# #json #azure #сериализация
Вопрос:
У меня есть POCO, который так же прост, как
public partial class Member
{
public int ID { get; set; }
[Required]
[StringLength(100)]
public string MemberId { get; set; }
public DateTime CreatedOn { get; set; }
[Required]
[StringLength(100)]
public string FirstName { get; set; }}
и команда добавления, которая выглядит следующим образом
public class AddMemberCommand : ICommand
{
public AddMemberCommand(Member member )
{
ID = Guid.NewGuid();
MemberData = member;
}
public Member MemberData { get; private set; }
public Guid ID { get; }
public string CommandName
{
get { return "AddMemberCommand"; }
}
}
который наследует от
public interface ICommand
{
/// <summary>
/// Gets the command identifier.
/// </summary>
Guid ID { get; }
string CommandName { get; }
}
Теперь я отправляю этот код методу, который инициализирует класс настройки сериализации Newton Json с некоторыми параметрами для возврата объекта. Сериализатор выглядит следующим образом
public class JsonTextSerializer
{
private readonly JsonSerializer _serializer;
public JsonTextSerializer()
{
_serializer = JsonSerializer.Create(new JsonSerializerSettings
{
TypeNameHandling = TypeNameHandling.All,
ReferenceLoopHandling = ReferenceLoopHandling.Ignore,
TypeNameAssemblyFormat = FormatterAssemblyStyle.Simple
});
}
public void Serialize(TextWriter writer, object graph)
{
var jsonWriter = new JsonTextWriter(writer);
jsonWriter.Formatting = Formatting.Indented;
_serializer.Serialize(jsonWriter, graph);
writer.Flush();
}
public object Deserialize(TextReader reader)
{
var jsonReader = new JsonTextReader(reader);
try
{
return this._serializer.Deserialize(jsonReader);
}
catch (JsonSerializationException e)
{
// Wrap in a standard .NET exception.
throw new SerializationException(e.Message, e);
}
}
}
Сериализатор используется для преобразования команды в полезную нагрузку для посреднического сообщения, как показано ниже
private BrokeredMessage CreateMessage(POCOS.Member member)
{
var serializer = new JsonTextSerializer();
var command = new AddMemberCommand(member);
var stream = new MemoryStream();
var writer = new StreamWriter(stream);
serializer.Serialize(writer, command);
stream.Position = 0;
BrokeredMessage message = new BrokeredMessage(stream, true);
return message;
}
и есть другой метод, который получает метод
private POCOS.Member GetPocoFromMessage(BrokeredMessage message)
{
ITextSerializer serializer = new JsonTextSerializer();
AddMemberCommand command;
using (var stream = message.GetBody<Stream>())
{
using (var reader = new StreamReader(stream))
{
var payload = serializer.Deserialize(reader);
command = payload as AddMemberCommand;
}
}
return command.MemberData;
}
Проблема заключается в десериализации, некоторые свойства (ID, CommandName) заполняются значением, за исключением MemberData, которое равно null.
Я могу прочитать поток (выполнив reader.ReadToEnd()) и увидеть, что он был передан по проводам, но Json не может десериализовать весь свой объект
Одно время я также думал, что, возможно, он выбирает только поля в интерфейсе, но это не так
Ответ №1:
У вашего MemberData
свойства есть собственный установщик. Поскольку сериализатору необходим доступ к свойству извне, этот установщик должен быть общедоступным.