在這篇簡短的文章中,我將解釋在使用JSON傳輸資料時,為什麼浮點數或大十進位制值應表示為字串 。
long型別引發的詭異情況
長話短說,同事在利用swagger對接後端API時,詭異的發現swaggerUI中顯示的json屬性值並不是api返回的值。
[HttpGet]
public IActionResult QueryAsync()
{
var testJson = new
{
Id = 123123126964992223,
Profile = "Please attention on Id",
};
return new JsonResult(testJson);
}
該API在swagger輸出:
{"Id": 123123126964992220,
"Profile": "Please attention on Id"}
進一步從Chrome->[Network]->[Preview]、[Response payload]觀察到該long屬性值的差異。
直接給結論:部分long型別值(最大值263-1)會超過Javascript的最大安全Number(253-1), 瀏覽器/前端 使用JSON.parse(123123126964992223)將不再保證準確性。
將JSON中的數字值作為字串傳輸的是為了消除傳輸中的精度丟失或歧義性。
JSON規範中未給數字指定精度,JSON解析器會自由選擇合適的數值精度。如果您的應用程式具有特定的精度要求,那麼在不同的JSON解析器可能不能正確表達精度。
另外部分long型別值(最大值263-1)會超過Javascript的最大安全Number(253 -1), 前端json反序列化時也會出現錯誤。
stackoverflow有個解釋很贊:
覆寫.NET Core序列化框架,將long轉化為string
針對NewtonsoftJson編寫BigIntJsonConvert
public class BigIntJsonConverter : JsonConverter<long>
{
public override long ReadJson(JsonReader reader, Type objectType, [AllowNull] long existingValue, bool hasExistingValue, JsonSerializer serializer)
{
var flag = long.TryParse(reader.ReadAsString(), out long num);
return flag == true ? num : 0;
}
public override void WriteJson(JsonWriter writer, [AllowNull] long value, JsonSerializer serializer)
{
writer.WriteValue(value.ToString());
}
}
// 擷取自Startup.cs ConfigureServices函式
context.Services.AddMvc().AddNewtonsoftJson(options =>
{
options.SerializerSettings.Converters.Add(new BigIntJsonConverter());
});