c#: Newtonsoft.Json 高级用法一(不创建类,动态解析和构造json、JObject/JArray)

本文主要是介绍c#: Newtonsoft.Json 高级用法一(不创建类,动态解析和构造json、JObject/JArray),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

环境:

  • .net core3.1
  • vs2019
  • Newtonsoft.Json 12.0.3

关于newtonsoft.json的使用常见问题参考:
《c#:序列化json常见问题及处理方法》
《c#:关于NewtonsoftJson序列化和Grpc序列化的冲突问题》

问题:如何在不新建类的情况下动态解析json字符串?

如下面:

{"Color": {"Red": 0.8,"Blue": 1.2},"Category": {"Better": [ 99, 98, 100, 96 ],"Normal": [ 80, 75, 84 ]}
}

问: 为什么不能使用指定的类进行反序列化?
答: json中的格式不能完全确定,可能是为将来保留,或者是供别人使用,而自己只负责读写其中的一块,如:“$.Color”

下面的代码就演示如何动态解析这个json字符串:

static void TestReadJson()
{var json = @"
{""Color"": {""Red"": 0.8,""Blue"": 1.2},""Category"": {""Better"": [ 99, 98, 100, 96 ],""Normal"": [ 80, 75, 84 ]}
}";JObject obj = Newtonsoft.Json.JsonConvert.DeserializeObject<JObject>(json);//color_red=0.8var color_red = obj["Color"]["Red"].Value<decimal>();//color_blue=1.2var color_blue = obj["Color"]["Blue"].Value<double>();//other ==null truevar other = obj["Color:NotExsit"];Console.WriteLine($"other==null => {other == null}");//d = 99var d = obj["Category"]["Better"][0].Value<int>();//遍历所有属性var propNameList = new List<string>();var propValueList = new List<double>();var props = (obj["Color"] as JObject).Properties();foreach (var prop in props){propNameList.Add(prop.Name);propValueList.Add(prop.Value.Value<double>());}//propNameList=Red,Blue//propValueList=0.8,1.2//遍历数组所有元素var values = new List<int>();var arr = obj["Category"]["Better"] as JArray;foreach (var item in arr){values.Add(item.Value<int>());}//values=99, 98, 100, 96
}

问题: 如何在不新建类的情况下动态组装json?

上面只说了动态解析,那么还有动态组装json的情况,如下面的json:

{"Color": {"Red": 0.8,"Blue": 1.2},"Category": {"Better": [ 99, 98, 100, 96 ],"Normal": [ 80, 75, 84 ],"Bad": [undefined,null]}
}

看下面的代码:

static void TestCreateJson()
{/* 手动组装json{"Color": {"Red": 0.8,"Blue": 1.2},"Category": {"Better": [ 99, 98, 100, 96 ],"Normal": [ 80, 75, 84 ],"Bad": [undefined,null]}}*///写入JObject//创建json字符串根节点var obj = new JObject(); //{}//Color节点obj["Color"] = new JObject();// { "Color" : {} }obj["Color"]["Red"] = 0.8;// { "Color" : { "Red" : 0.8  } }obj["Color"]["Blue"] = 1.2;// { "Color" : { "Red" : 0.8 , "Blue" : 1.2 } }//Category节点obj["Category"] = new JObject();// { "Color" : { "Red" : 0.8 , "Blue" : 1.2 }, "Category": {} }//Category->Better数组obj["Category"]["Better"] = new JArray();(obj["Category"]["Better"] as JArray).Add(99);(obj["Category"]["Better"] as JArray).Add(98);(obj["Category"]["Better"] as JArray).Add(100);(obj["Category"]["Better"] as JArray).Add(96); //{ "Color" : { "Red" : 0.8 , "Blue" : 1.2 }, "Category": { “Better”: [99, 98, 100, 96] } }//Category->Normal数组obj["Category"]["Normal"] = new JArray();(obj["Category"]["Normal"] as JArray).Add(80);(obj["Category"]["Normal"] as JArray).Add(75);(obj["Category"]["Normal"] as JArray).Add(84);//{ "Color" : { "Red" : 0.8 , "Blue" : 1.2 }, "Category": { “Better”: [99, 98, 100, 96], “Normal”: [80, 75, 84] } }//Category->Bad数组obj["Category"]["Bad"] = new JArray();(obj["Category"]["Bad"] as JArray).Add(JValue.CreateUndefined());(obj["Category"]["Bad"] as JArray).Add(JValue.CreateNull());//{ "Color" : { "Red" : 0.8 , "Blue" : 1.2 }, "Category": { “Better”: [99, 98, 100, 96], “Normal”: [80, 75, 84], "Bad": [ undefined, null ] } }//输出jsonvar jsonStr = obj.ToString();
}

问题: 如何在不新建类的情况下改写json?

既然上面有了动态解析和组装,那么就有改写部分json的需求,看如下代码:

static void TestUpdateJson()
{var json = @"
{""Color"": {""Red"": 0.8,""Blue"": 1.2},""Category"": {""Better"": [ 99, 98, 100, 96 ],""Normal"": [ 80, 75, 84 ],""Bad"": [undefined,null]}
}";var obj = Newtonsoft.Json.JsonConvert.DeserializeObject<JObject>(json);obj["Color"] = "Red,Blue";obj["Category"]["Better"][0] = 100;obj["Category"]["Normal"][0] = new JObject();json = obj.ToString();/*{"Color": "Red,Blue","Category": {"Better": [100,98,100,96],"Normal": [{},75,84],"Bad": [null,null]}}*/
}

问题: 如何动态删除json里的内容?

直接看代码:

static void TestDeleteJson()
{var json = @"
{""Color"": {""Red"": 0.8,""Blue"": 1.2},""Category"": {""Better"": [ 99, 98, 100, 96 ],""Normal"": [ 80, 75, 84 ],""Bad"": [undefined,null]}
}";//移除属性和数组元素var obj = Newtonsoft.Json.JsonConvert.DeserializeObject<JObject>(json);(obj["Color"] as JObject).Remove("Red");(obj["Category"] as JObject).Remove("Better");obj["Category"]["Normal"][0].Remove();json = obj.ToString();/*{"Color": {"Blue": 1.2},"Category": {"Normal": [80,75,84],"Bad": [null,null]}}*/
}

问题:用考虑json文件中的注释吗?

用!!!
因为,我们现在是将json读取到JArray/JObject,Newtonsoft.Json 会将注释一并读取到结构,所以我们要手动排除注释的干扰,具体可参考:
《Newtonsoft.Json 反序列化注意点》

这篇关于c#: Newtonsoft.Json 高级用法一(不创建类,动态解析和构造json、JObject/JArray)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/992564

相关文章

Java中Redisson 的原理深度解析

《Java中Redisson的原理深度解析》Redisson是一个高性能的Redis客户端,它通过将Redis数据结构映射为Java对象和分布式对象,实现了在Java应用中方便地使用Redis,本文... 目录前言一、核心设计理念二、核心架构与通信层1. 基于 Netty 的异步非阻塞通信2. 编解码器三、

JDK21对虚拟线程的几种用法实践指南

《JDK21对虚拟线程的几种用法实践指南》虚拟线程是Java中的一种轻量级线程,由JVM管理,特别适合于I/O密集型任务,:本文主要介绍JDK21对虚拟线程的几种用法,文中通过代码介绍的非常详细,... 目录一、参考官方文档二、什么是虚拟线程三、几种用法1、Thread.ofVirtual().start(

Java HashMap的底层实现原理深度解析

《JavaHashMap的底层实现原理深度解析》HashMap基于数组+链表+红黑树结构,通过哈希算法和扩容机制优化性能,负载因子与树化阈值平衡效率,是Java开发必备的高效数据结构,本文给大家介绍... 目录一、概述:HashMap的宏观结构二、核心数据结构解析1. 数组(桶数组)2. 链表节点(Node

Java 虚拟线程的创建与使用深度解析

《Java虚拟线程的创建与使用深度解析》虚拟线程是Java19中以预览特性形式引入,Java21起正式发布的轻量级线程,本文给大家介绍Java虚拟线程的创建与使用,感兴趣的朋友一起看看吧... 目录一、虚拟线程简介1.1 什么是虚拟线程?1.2 为什么需要虚拟线程?二、虚拟线程与平台线程对比代码对比示例:三

从基础到高级详解Go语言中错误处理的实践指南

《从基础到高级详解Go语言中错误处理的实践指南》Go语言采用了一种独特而明确的错误处理哲学,与其他主流编程语言形成鲜明对比,本文将为大家详细介绍Go语言中错误处理详细方法,希望对大家有所帮助... 目录1 Go 错误处理哲学与核心机制1.1 错误接口设计1.2 错误与异常的区别2 错误创建与检查2.1 基础

k8s按需创建PV和使用PVC详解

《k8s按需创建PV和使用PVC详解》Kubernetes中,PV和PVC用于管理持久存储,StorageClass实现动态PV分配,PVC声明存储需求并绑定PV,通过kubectl验证状态,注意回收... 目录1.按需创建 PV(使用 StorageClass)创建 StorageClass2.创建 PV

一文解析C#中的StringSplitOptions枚举

《一文解析C#中的StringSplitOptions枚举》StringSplitOptions是C#中的一个枚举类型,用于控制string.Split()方法分割字符串时的行为,核心作用是处理分割后... 目录C#的StringSplitOptions枚举1.StringSplitOptions枚举的常用

Python函数作用域与闭包举例深度解析

《Python函数作用域与闭包举例深度解析》Python函数的作用域规则和闭包是编程中的关键概念,它们决定了变量的访问和生命周期,:本文主要介绍Python函数作用域与闭包的相关资料,文中通过代码... 目录1. 基础作用域访问示例1:访问全局变量示例2:访问外层函数变量2. 闭包基础示例3:简单闭包示例4

MyBatis延迟加载与多级缓存全解析

《MyBatis延迟加载与多级缓存全解析》文章介绍MyBatis的延迟加载与多级缓存机制,延迟加载按需加载关联数据提升性能,一级缓存会话级默认开启,二级缓存工厂级支持跨会话共享,增删改操作会清空对应缓... 目录MyBATis延迟加载策略一对多示例一对多示例MyBatis框架的缓存一级缓存二级缓存MyBat

Linux创建服务使用systemctl管理详解

《Linux创建服务使用systemctl管理详解》文章指导在Linux中创建systemd服务,设置文件权限为所有者读写、其他只读,重新加载配置,启动服务并检查状态,确保服务正常运行,关键步骤包括权... 目录创建服务 /usr/lib/systemd/system/设置服务文件权限:所有者读写js,其他