Это похоже на ошибку в сериализаторе. В. NET Core 3.1 есть некоторый код в CreateDerivedEnumerableInstance(ref ReadStack state, JsonPropertyInfo collectionPropertyInfo, IList sourceList)
для создания стека из десериализованного списка:
else if (instance is Stack<TDeclaredProperty> instanceOfStack)
{
foreach (TDeclaredProperty item in sourceList)
{
instanceOfStack.Push(item);
}
return instanceOfStack;
}
Однако он толкает их в неправильном порядке. Таким образом, custom JsonConverter<Stack<T>>
потребуется для правильной десериализации Stack<T>
. Кроме того, JsonConverterFactory
может использоваться для изготовления соответствующего преобразователя для каждого типа стека Stack<T>
:
public class StackConverterFactory : JsonConverterFactory
{
public override bool CanConvert(Type typeToConvert)
{
return GetStackItemType(typeToConvert) != null;
}
public override JsonConverter CreateConverter(Type typeToConvert, JsonSerializerOptions options)
{
var itemType = GetStackItemType(typeToConvert);
var converterType = typeof(StackConverter<,>).MakeGenericType(typeToConvert, itemType);
return (JsonConverter)Activator.CreateInstance(converterType);
}
static Type GetStackItemType(Type type)
{
while (type != null)
{
if (type.IsGenericType)
{
var genType = type.GetGenericTypeDefinition();
if (genType == typeof(Stack<>))
return type.GetGenericArguments()[0];
}
type = type.BaseType;
}
return null;
}
}
public class StackConverter<TItem> : StackConverter<Stack<TItem>, TItem>
{
}
public class StackConverter<TStack, TItem> : JsonConverter<TStack> where TStack : Stack<TItem>, new()
{
public override TStack Read(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options)
{
var list = JsonSerializer.Deserialize<List<TItem>>(ref reader, options);
if (list == null)
return null;
var stack = typeToConvert == typeof(Stack<TItem>) ? (TStack)new Stack<TItem>(list.Count) : new TStack();
for (int i = list.Count - 1; i >= 0; i--)
stack.Push(list[i]);
return stack;
}
public override void Write(Utf8JsonWriter writer, TStack value, JsonSerializerOptions options)
{
writer.WriteStartArray();
foreach (var item in value)
JsonSerializer.Serialize(writer, item, options);
writer.WriteEndArray();
}
}
Затем используйте его в JsonSerializerOptions
следующим образом:
var stack = new Stack<int>(new [] { 1, 2, 3 });
var options = new JsonSerializerOptions
{
Converters = { new StackConverterFactory() },
};
var json = JsonSerializer.Serialize(stack, options);
var stack2 = JsonSerializer.Deserialize<Stack<int>>(json, options);
var json2 = JsonSerializer.Serialize(stack2, options);
Assert.IsTrue(stack.SequenceEqual(stack2)); // Passes
Assert.IsTrue(json == json2); // Passes
Конвертер также может быть применен непосредственно к некоторой модели данных с использованием JsonConverterAttribute
public class Model
{
[JsonConverter(typeof(StackConverter<int>))]
public Stack<int> Stack { get; set; }
}
Демо-скрипта здесь .
Обновление : выглядит, что круговое отключение Stack<T>
не будет встроено в JsonSerializer
. См. (Де) сериализацию стеков с помощью JsonSerializer, если в оба конца # 41887 (Закрыто) :
Мы не должны этого делать. Не существует стандарта, на какой стороне следует переворачивать элементы (сериализацию или десериализацию) для того, чтобы совершить круговую поездку, так что это не стартер в качестве кандидата на критические изменения. Текущее поведение совместимо с поведением Newtonsoft. Json.
Существует рабочий элемент, обеспечивающий примерный конвертер, показывающий, как использовать обходные пути в документах JSON, которые, как мне кажется, должно быть достаточно для решения этой проблемы: Dotnet / документы # 16690 . Вот как мог бы выглядеть этот конвертер - dotnet / docs # 16225 (комментарий) .