System.Text.Json类库进行json转化时ValueKind:Object问题

当你的使用的Json库是System.Text.Json,而不是Newtonsoft.Json库的时候,你可能遇到以下问题及其解决办法。通常的解决办法是进行一些对应的配置。此外就需要根据情况使用自定义转换器实现你的需求。以下是通常遇到的使用自定义转换器解决的例子:

Q1.当遇到System.InvalidCastException:"Unable to cast object of type 'System.Text.Json.JsonElement' to type 'System.Int32'."这种错误的时候,是由于System.Text.Json类库进行json转化时,没有明确类型的基础类型导致的。通常可能是某一个属性类型使用object情况。

A1:解决办法是写一个 对象Json转换器 类,如下所示:

csharp 复制代码
  /// <summary>
 /// 对象Json转化器
 /// </summary>
 public class ObjectJsonConverter : JsonConverter<Object>
 {
     public override object? Read(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options)
     {
         switch (reader.TokenType)
         {
             case JsonTokenType.String:
                 if (DateTime.TryParse(reader.GetString(), out DateTime dateTime))
                     return dateTime;
                 return reader.GetString();
             case JsonTokenType.Number:
                 if (reader.TryGetInt32(out int intValue))
                     return intValue;
                 else if (reader.TryGetDouble(out double doubleValue))
                     return doubleValue;
                 else if (reader.TryGetDecimal(out decimal decimalValue))
                     return decimalValue;
                 throw new JsonException("Unsupported number format");
             case JsonTokenType.True:
                 return true;
             case JsonTokenType.False:
                 return false;
             case JsonTokenType.Null:
                 return null;
             case JsonTokenType.StartObject:
                 return JsonSerializer.Deserialize<JsonElement>(ref reader, options);
             case JsonTokenType.StartArray:
                 return JsonSerializer.Deserialize<JsonElement[]>(ref reader, options);
             default:
                 throw new JsonException($"Unexpected token type: {reader.TokenType}");
         }
     }

     public override void Write(Utf8JsonWriter writer, object value, JsonSerializerOptions options)
     {
         if (value == null) writer.WriteNullValue();
         else if ((value is string || value is DateTime || value is Guid)) writer.WriteStringValue(value.ToString());
         else if (value is int intValue) writer.WriteNumberValue(intValue);
         else if (value is double doubleValue) writer.WriteNumberValue(doubleValue);
         else if (value is decimal decimalValue) writer.WriteNumberValue(decimalValue);
         else if (value is char charValue) writer.WriteNumberValue(charValue);
         else if (value is bool boolValue) writer.WriteBooleanValue(boolValue);
         else if (value is JsonElement jsonElement) jsonElement.WriteTo(writer);
         else if (value is JsonElement[] jsonElements)
         {
             writer.WriteStartArray();
             foreach (var element in jsonElements)
             {
                 element.WriteTo(writer);
             }
             writer.WriteEndArray();
         }
         else JsonSerializer.Serialize(writer, value, value.GetType(), options);
     }
 }

使用如下:

csharp 复制代码
builder.Services.AddJsonOptions(options =>
{
    // 添加Object格式化转换器
    options.JsonSerializerOptions.Converters.Add(new ObjectJsonConverter());
});

Q2:在webapi接口返回结果中,时间属性的值返回的形式如:"2024-01-01T10:00:00",而我们想要的是不带T的格式,比如"2024-01-01 10:00:00"这种。

A2:需要定义一个时间Json转换器DateTimeJsonConverter,代码如下:

csharp 复制代码
 /// <summary>
	/// 设置Json默认DateTime格式化
	/// </summary>
	public class DateTimeJsonConverter : JsonConverter<DateTime>
 {
     private readonly string _format;
     public DateTimeJsonConverter(string format)
     {
         _format = format;
     }
     public override void Write(Utf8JsonWriter writer, DateTime dateTime, JsonSerializerOptions options)
     {
         writer.WriteStringValue(dateTime.ToString(_format));
     }
     public override DateTime Read(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options)
     {
         if (reader.TokenType == JsonTokenType.String)
         {
             if (DateTime.TryParse(reader.GetString(), out DateTime dateTime))
                 return dateTime;
         }
         return reader.GetDateTime();
     }
 }

使用如下:

csharp 复制代码
builder.Services.AddJsonOptions(options =>
{
     // 添加时间格式化转换器
 options.JsonSerializerOptions.Converters.Add(new DateTimeJsonConverter("yyyy-MM-dd HH:mm:ss"));
});

通常完整一点的配置如下:

csharp 复制代码
builder.Services.AddJsonOptions(options =>
{
    // 设置编码格式
options.JsonSerializerOptions.Encoder = JavaScriptEncoder.Create(UnicodeRanges.All);
// 是否格式化文本
options.JsonSerializerOptions.WriteIndented = true;
//不区分大小写
options.JsonSerializerOptions.PropertyNameCaseInsensitive = true;//不区分大小写
// 字段采用驼峰式命名
options.JsonSerializerOptions.PropertyNamingPolicy = JsonNamingPolicy.CamelCase;
// 忽略null值
//options.JsonSerializerOptions.DefaultIgnoreCondition = JsonIgnoreCondition.WhenWritingNull;
// 忽略只读字段
//options.JsonSerializerOptions.IgnoreReadOnlyProperties = true;
// 对字典的键进行驼峰命名
options.JsonSerializerOptions.DictionaryKeyPolicy = JsonNamingPolicy.CamelCase;
//不允许有注释的不标准json
options.JsonSerializerOptions.ReadCommentHandling = JsonCommentHandling.Disallow;
// 允许属性值末尾存在逗号
options.JsonSerializerOptions.AllowTrailingCommas = true;
// 处理循环引用类型
options.JsonSerializerOptions.ReferenceHandler = ReferenceHandler.IgnoreCycles;
//枚举类型转string配置(避免转int)
//options.JsonSerializerOptions.Converters.Add(new JsonStringEnumConverter());
// 添加时间格式化转换器
options.JsonSerializerOptions.Converters.Add(new DateTimeJsonConverter("yyyy-MM-dd HH:mm:ss"));
// 添加Object格式化转换器
options.JsonSerializerOptions.Converters.Add(new ObjectJsonConverter());
});

Q3:遇到枚举的属性,如果想在接口返回的时候不返回数字,而想返回字符串的情况

A3:在配置的时候进行如下配置即可:

csharp 复制代码
builder.Services.AddJsonOptions(options =>
{
     //枚举类型转string配置(避免转int)
	options.JsonSerializerOptions.Converters.Add(new JsonStringEnumConverter());
});

此外在枚举的地方添加[JsonConverter(typeof(JsonStringEnumConverter))]特性即可返回字符串,代码如下:

csharp 复制代码
 public class Sample
 {
     public int Id { get; set; }
     public object  Value { get; set; }
     [JsonConverter(typeof(JsonStringEnumConverter))]
     public SampleType Type { get; set; }
 }

 public enum SampleType
 {
     AAAA,
     BBBB,
     CCCC
 }

如果有不够的地方,自行添加。

相关推荐
追逐时光者2 小时前
C#/.NET/.NET Core技术前沿周刊 | 第 41 期(2025年6.1-6.8)
后端·.net
追逐时光者2 小时前
不写一行代码 .NET 使用 FluentCMS 快速构建现代化内容管理系统(CMS)
后端·.net·cms
ghost1432 小时前
C#学习第29天:表达式树(Expression Trees)
开发语言·c#
百锦再10 小时前
Razor编程中@Helper的用法大全
.net·web·blazor·tag·core·razor·helper
安木夕12 小时前
C#-Visual Studio宇宙第一IDE使用实践
前端·c#·.net
gregmankiw15 小时前
C#调用Rust动态链接库DLL的案例
开发语言·rust·c#
阿蒙Amon16 小时前
06. C#入门系列【自定义类型】:从青铜到王者的进阶之路
开发语言·c#
o0向阳而生0o18 小时前
65、.NET 中DllImport的用途
.net·非托管·dllimport
喵叔哟18 小时前
25.【.NET8 实战--孢子记账--从单体到微服务--转向微服务】--单体转微服务--用户服务接口
微服务·架构·.net
钢铁男儿19 小时前
C# 表达式和运算符(表达式和字面量)
开发语言·c#