当前位置: 首页 > news >正文

C#中Json序列化的进阶用法

本文所有json序列化,都使用的Newtonsoft.Json包

1 JsonIgnore

在 Newtonsoft.Json 中,如果你不想将某些属性转换为 JSON 字符串,可以使用多种方法来实现。以下是几种常见的方法:

1.1 使用 [JsonIgnore] 特性

[JsonIgnore] 特性可以直接忽略某个属性,使其不会被序列化为 JSON。

using Newtonsoft.Json;public class MyClass
{public int Id { get; set; }public string Name { get; set; }[JsonIgnore]public string Password { get; set; }
}class Program
{static void Main(){var obj = new MyClass { Id = 1, Name = "John", Password = "secret" };string json = JsonConvert.SerializeObject(obj);Console.WriteLine(json);}
}

输出:

{"Id":1,"Name":"John"}

1.2 使用 [JsonProperty] 特性并设置 DefaultValueHandling

你可以使用 [JsonProperty] 特性并设置 DefaultValueHandlingIgnore,这样当属性的值为默认值时,它不会被序列化为 JSON。

using Newtonsoft.Json;public class MyClass
{public int Id { get; set; }public string Name { get; set; }[JsonProperty(DefaultValueHandling = DefaultValueHandling.Ignore)]public string Password { get; set; }
}class Program
{static void Main(){var obj = new MyClass { Id = 1, Name = "John", Password = null };string json = JsonConvert.SerializeObject(obj);Console.WriteLine(json);}
}

输出:

{"Id":1,"Name":"John"}

2 自定义解析方式

在使用 JsonConvert.DeserializeObject 解析 JSON 数据时,如果某个属性需要以特定的方式进行解析,可以通过自定义 JsonConverter 来实现。JsonConverter 允许你定义如何将 JSON 数据转换为 .NET 对象,或者将 .NET 对象转换为 JSON 数据。

以下是一个示例,展示如何为特定属性创建自定义的 JsonConverter

2.1 创建自定义的 JsonConverter

假设你有一个类 MyClass,其中有一个属性 MyProperty,你希望以特定的方式解析这个属性:

public class MyClass
{public string MyProperty { get; set; }
}

你可以创建一个自定义的 JsonConverter 来处理 MyProperty

public class MyPropertyConverter : JsonConverter<string>
{public override string ReadJson(JsonReader reader, Type objectType, string existingValue, bool hasExistingValue, JsonSerializer serializer){// 读取 JSON 数据var jsonObject = JObject.Load(reader);// 自定义解析逻辑if (jsonObject["customField"] != null){return jsonObject["customField"].ToString();}return string.Empty;}public override void WriteJson(JsonWriter writer, string value, JsonSerializer serializer){// 自定义序列化逻辑writer.WriteValue(value);}
}

2.2 在属性上应用自定义的 JsonConverter

你可以在 MyClassMyProperty 属性上使用 [JsonConverter] 特性来指定自定义的 JsonConverter

public class MyClass
{[JsonConverter(typeof(MyPropertyConverter))]public string MyProperty { get; set; }
}

2.3 使用 JsonConvert.DeserializeObject 进行解析

现在,当你使用 JsonConvert.DeserializeObject 解析 JSON 数据时,MyProperty 属性将按照你定义的自定义逻辑进行解析:

string json = "{ \"MyProperty\": { \"customField\": \"customValue\" } }";
MyClass myObject = JsonConvert.DeserializeObject<MyClass>(json);Console.WriteLine(myObject.MyProperty);  // 输出: customValue

2.4 总结

通过创建自定义的 JsonConverter 并在属性上应用它,你可以灵活地控制 JSON 数据的解析方式。这种方式适用于需要对特定属性进行特殊处理的场景。

3 进阶用法-动态读取,结合反射解析json

一般来说,添加了引用的内容,Newtonsoft.Json都可以直接解析,包括Type

动态读取的dll,如果放在了和执行文件同一目录下也可以正常解析,但是一旦脱离这个目录,json序列化就不行了,找不到对应的内容,尝试了很多方式都不行,结合第二部分,可以使用反射实现。

这种方式有个前提条件,目标dll需要在执行解析之前被读取,这样读取到的资源会保存到AppDomain.CurrentDomain中去,才能进行解析。

Assembly.LoadFrom(dllFile);
/// <summary>
/// 涉及到Type的序列化
/// 并且插件的程序集是程序显式读取的
/// 默认的解析方式读取不到插件对应的dll
/// 需要做自定义解析
/// </summary>
public class TypeJsonConverter : JsonConverter<Type>
{/// <summary>/// 自定义该属性的解析方式/// </summary>public override Type ReadJson(JsonReader reader, Type objectType, Type existingValue, bool hasExistingValue, JsonSerializer serializer){// 自定义解析逻辑if (reader.Value != null){//在调用该程序之前,需要确保插件的dll已经被显式读取,也就是已经存在于主程序集的上下文之中var typeValue = AppDomain.CurrentDomain.GetAssemblies().SelectMany(a => a.GetTypes()).Where(plu => plu.AssemblyQualifiedName.Equals(reader.Value)).FirstOrDefault();return typeValue;}return null;}/// <summary>/// 自定义该属性的写入方式/// </summary>public override void WriteJson(JsonWriter writer, Type value, JsonSerializer serializer){// 自定义序列化逻辑writer.WriteValue(value.AssemblyQualifiedName);}
}

然后标记属性

/// <summary>
/// 插件,标记自定义解析方式,否则会解析失败
/// 目前先这样处理
/// </summary>
[JsonConverter(typeof(TypeJsonConverter))]
public Type PluginType { get; set; }

http://www.mrgr.cn/news/46370.html

相关文章:

  • Github 2025-01-15 C开源项目日报 Top10
  • 一文通透OpenVLA及其源码剖析——基于Prismatic VLM(SigLIP、DinoV2、Llama 2)及离散化动作预测
  • 【前端】自学基础算法 -- 24.动态规划-变态青蛙蛙跳台阶
  • c语言-----常识问题
  • 【面试题】技术场景 5、日志采集ELK
  • 如何将 sqlserver 数据迁移到 mysql
  • [投稿优惠|稳定检索]2024年电子器件与机械工程、材料国际会议(EDMEM 2024)
  • 系统架构师备考记忆不太清楚的点-信息系统-需求分析
  • 10.9今日错题解析(软考)
  • 低代码开发平台应该归属哪个部门管理?
  • 2003 -Can‘t connect to MySQL server on‘192.168.‘(10060 “Unknown error“
  • Maven 三种项目打包方式:POM、JAR 和 WAR 的区别详解
  • 基于开元鸿蒙(OpenHarmony)的【智能药房与药品管理综合应用系统】
  • 微服务实战——登录(普通登录、社交登录、SSO单点登录)
  • 新硬盘第一次使用需要怎样做?
  • JDK17安装教程
  • 超详解C++类与对象(中)
  • 学习内容的记录学习方向
  • 关于halcon深度图tiff生成点云文件
  • 【新品发布】数字能源EMS管理再掀新篇章
  • 环保数采仪下噪音在线监测,打造声环境精准管控解决方案
  • 每日一练:最长回文子串
  • 【VUE】虚拟DOM真的比真实DOM性能好吗
  • Synchronized的缺陷
  • Ubuntu下Typora的安装与配置激活
  • RTSP RTP RTCP SDP基础知识