问题描述
我正在使用BigInteger
将System.Text.Json
序列化为JSON:
JsonSerializer.Serialize(new {foo = new BigInteger(ulong.MaxValue) + 1})
这将产生以下输出:
{"foo":{"IsPowerOfTwo":true,"IsZero":false,"IsOne":false,"IsEven":true,"Sign":1}}
如果我添加一个将BigInteger
值转换为ulong
的转换器,则它当然会失败,因为BigInteger
值太大:
var options = new JsonSerializerOptions();
options.Converters.Add(new BigIntegerConverter());
JsonSerializer.Serialize(new {foo = new BigInteger(ulong.MaxValue) + 1},options);
这是转换器:
public class BigIntegerConverter : JsonConverter<BigInteger>
{
public override BigInteger Read(ref Utf8JsonReader reader,Type typeToConvert,JsonSerializerOptions options) => throw new NotImplementedException();
public override void Write(Utf8JsonWriter writer,BigInteger value,JsonSerializerOptions options) => writer.WriteNumberValue((ulong)value);
}
我想要的输出是:
{"foo":18446744073709551616}
我知道可以在Json.NET中使用JsonWriter.WriteRawValue()
来实现,但是我仅限于使用System.Text.Json
。
有什么方法可以做到,而无需手动破解序列化的字符串?
解决方法
为BigInteger
写一个转换器有点尴尬,因为您注意到,Utf8JsonReader
和Utf8JsonWriter
在.NET 5中不提供读写原始JSON的能力。
JsonDocument
,但是确实通过RootElement.GetRawText()
提供对原始JSON的访问,因此您可以通过读取和写入中间文档来编写转换器,如下所示:
public class BigIntegerConverter : JsonConverter<BigInteger>
{
public override BigInteger Read(ref Utf8JsonReader reader,Type typeToConvert,JsonSerializerOptions options)
{
if (reader.TokenType != JsonTokenType.Number)
throw new JsonException(string.Format("Found token {0} but expected token {1}",reader.TokenType,JsonTokenType.Number ));
using var doc = JsonDocument.ParseValue(ref reader);
return BigInteger.Parse(doc.RootElement.GetRawText(),NumberFormatInfo.InvariantInfo);
}
public override void Write(Utf8JsonWriter writer,BigInteger value,JsonSerializerOptions options)
{
var s = value.ToString(NumberFormatInfo.InvariantInfo);
using var doc = JsonDocument.Parse(s);
doc.WriteTo(writer);
}
}
演示小提琴here。