首页 程序笔记 C++中对JSON格式的解析以及序列化和反序列化

C++中对JSON格式的解析以及序列化和反序列化

前言

JSON是很常用的数据格式,在C++中我们可以使用nlohmann json做序列化和反序列化。
nlohmann json是一个跨平台的开源的C++ JSON解析库,只需要在项目中#include json.hpp文件就可以使用了。
github上面的文档非常详细,你也可以通过本文对nlohmann json有个基本了解,本文涵盖了日常编码中会用json库处理的大部分操作,如构造json对象、查找键值、序列化与反序列。
github地址: https://github.com/nlohmann/json

nlohmann json基础操作

由basic value创建json value

 using nlohmann::json;
 //方式一:
 json j1;
 j[name]=LeBorn Jame;//字符串
 j[number]=23;//整数
 j[man]=true;//布尔值
 j[children]={LeBorn Jr,Bryce Maximus,Zhuri};//数组
 j[behavior][funny]=gigigigigigi;//对象中元素值
 j[wife]={{name,Savannah Brinson},{man,false}};//对象
 
 //方式二
 json j2={
 {name,LeBorn Jame},
 {number,23},
  {man,true},
 {children,{LeBorn Jr,Bryce Maximus,Zhuri}},
 {behavior,{{funny,gigigigigigi}};
 {wife,{{name,Savannah Brinson},{man,false}}}
 };

由json value得到basic value

 using nlohmann::json;
 json j={
 {name,LeBorn Jame},
 {number,23},
 {children,{LeBorn Jr,Bryce Maximus,Zhuri}},
 {man,true},
 {wife,{name,Savannah Brinson}}
 };
 auto name=j[name].getstd::string();//方式一
 int number=0;
 j[number].get_to(number);//方式二
 auto childOne = j[children][0].getstd::string();//获取数组元素方式一,越界后不会抛出异常,直接中断
 auto childTwo=j[children].at(1).getstd::string();//越界后抛出异常
 auto wifeName=j[wife][name].getstd::string();
 
//bool man=j[man];//不推荐使用这种隐式转换的方式

像操作stl container一样操作json value

using nlohmann::json;
json animalArray={cat,dog};//定义一个数组类型的json对象
animalArray.push_back(pig);//添加元素
animalArray.emplace_back(duck);//C++11新方式添加元素,减少申请内存
//使用is_array()函数判断对象类型,使用empty函数判断数量是否为空
if(animalArray.is_array()  !animalArray.empty())
{
    auto size=animalArray.size();//使用size函数获取元素数量
    auto animalLast=animalArray.at([size-1]).getstd::string();
}

json animalObject={{kind,dog},{height,50}};//定义一个对象类型的json对象
animalObject.emplace({color,red});//插入元素
animalObject.erase(kind);//删除键值

//判断是否含有某个键值方式一
if(animalObject.contains(height))//通过contains函数判断是否包含某个key
{
    auto height=animalObject[height].getdouble();
}
//判断是否含有某个键值方式二
auto size=animalObject.count(height);//通过count函数计算某一个键的数量
if(size0)
{
    
}
//判断是否含有某个键值方式三
auto iter=animalObject.find(height);//通过find函数查找某个键的迭代器
if(iter!=animalObject.end()
{
    
}
//遍历输出键值方式一
for(auto item:animalObject.items())
{
    std::coutitem.key() item.value()std::endl;
}
//遍历输出键值方式2
for(auto iter=animalObject.begin();iter!=animalObject.end();++iter)
{
    std::coutiter-key() iter-value()std::endl;
}
//遍历输出键值方式3(C++17)
for(auto[key,value]:animalObject.keys())
{
    std::coutkey valuestd::endl;
}

序列化与反序列化(Serialization/Deserialization)

json value和string

using nlohmann::json;
//反序列化(Deserialization):_json/parse()
json j1={\kind\:\dog\,\height\:50}_json;//方式一,通过_json实现
json j2=R({kind:dog,height:50});//使用原生字符串关键字R来避免转移字符,但这一句并没有序列化,j2只保存字符串而已
json j3=json::parse(R({kind:dog,height:50}));//方式二,通过静态函数parse实现。
//使用 parse 函数时,如果字符串不符合 json 规范导致解析失败,默认会抛出异常,可以使用 try...catch 结构捕获异常进行处理;或者设置 parse 函数的第三个函数参数为 false,此时解析失败不会抛出异常,但解析得到的结果为 null。

//序列化(Serialization):dump(number),number为打印出的空格数
std::string animal1=j1.dump();//animal1值为{kind:dog,height:50}
std::string animal2=j1.dump(4);
//animal2值为
// {
//      kind:dog,
//      height:50,
// }

json value和streams

using nlohmann::json;
//反序列化(Deserialization):
json j1;
std::cinj1;

//序列化(Seriralization):
std::coutj1;

//上述操作适用于istream和ostream的子类,比如我们经常会用到的ifstream和ofstream
std::ifstream in(animals.json);//打开文件,关联到流in
json animals;
inanimals;
animals.at(0)[height]=60;//修改数据
in.close();

std::ofstream out(animal.json);//打开文件,关联到流out
outstd::setw(4)animals;//输出信息,std::setw(4)用于设置增加打印空格
out.close();

json value和自定义对象

在自定义对象命名空间中定义两个函数即可像basic value一样进行反序列化和序列化:from_json(const json j,T value)、to_json(json j,const T value)

//Animal.h文件
//定义Animal类,其包含两个成员变量kind与height
using nlohmann::json;
class Animal
{
    public:
        Animal(std::string kind,double height){
            this-kind=kind;
            this-height=height;
        }
    std::string kind;
    double height;
}
//定义from_json(const json j,T value)函数,用于序列化
void from_json(const json j,Animal animal)
{
    animal.kind=j[kind].getstd::string();
    animal.height=j[height].getdouble();
}

//定义to_json(json j,const T value)函数,用于反序列化
void to_json(json j,const Animal animal)
{
    j[kind]=animal.kind;
    j[height]=animal.height;
}

//main.cpp文件
int main()
{
    Animal animal{dog,50};
    nlohmnn::json j=animal;//像basic value一样将自定义对象赋值给json value
    j[height]=60;//修改数据
    Animal animalNew = j.getAnimal();//像basic value一样通过get函数获取值,将其值直接赋值给自定义对象
    std::coutanimal.height;//输出60
    return 0;
}

3

站心网

前言 JSON是很常用的数据格式,在C++中我们可以使用nlohmann json做序列化和反序列化。 nlohmann json是一..

为您推荐

JS 中彻底删除 JSON 对象组成的数组中的元素

在 JS 中,对于某个由 JSON 对象组成的数组,例如:var test = [{ "a": "1", "b": "2" }, { "a": "3", "b": "4" }, { "a": "5", "b": "6" }];如果我们想要删除其中的第二个json对象,应该怎么做呢?其实方法和操作数..

.NET 9 即将推出的功能Task.WhenEach

.NET 爱好者!我刚刚偶然发现了一个非常酷的新 PR,它被合并到 .NET 运行时存储库中,我想分享一个例子。希望您能为新的 .NET 版本大肆宣传!在即将到来的 .NET 9 版本中,我们预计会有一个名为 .它在这里让您的异步..

针对 Go 语言开发的 SQL 驱动模拟库

数据库交互是几乎所有应用程序不可或缺的一部分,开发者们常常需要对数据库进行各种操作,包括插入、更新、删除和查询等。然而,在开发过程中直接对真实数据库进行操作不仅耗时耗力,还可能带来数据一致性和安全性的..

DockerUI 中文可视化Docker管理工具使用示例

DockerUI 是由国内开发者打造的一款优秀的 Docker 可视化管理工具。该工具拥有简洁直观的UI界面,可以轻松进行Docker主机管理、集群管理,以及Docker任务的编排等操作。DockerUI不仅展示了资源利用率、系统信息和更..

Blazor的N种渲染模式原理和常见问题说明

我们从下面这幅图开始,下图显示了三种渲染模式,分别称之为静态SSR、交互式SSR(即之前的BlazorServer)、交互式CSR(即之前的BlazorWasm)。还有一种渲染模式BlazorHybrid,稍后说。一、先浅层理解一个图例静态SSR:经..

前端CSS常见的三种设计模式

CSS设计模式主要包括OOCSS、SMACSS和BEMCSS等。以下是对这些模式的具体介绍:OOCSS:面向对象的CSS,旨在编写高可复用、低耦合和高扩展的CSS代码。它将抽象(结构)和实现(样式)分离,抽离公共代码,以提高代码的..

WinToUSB | 把Windows塞进U盘里即插即用

不论是在外出差,还是在家临时办公,现在很多设备携带起来都不是那么方便,在这种情况下,有一个轻巧而高效的操作系统环境就显得格外关键。今天,要给大家介绍一款超级实用的便携式系统启动盘,凭借其独特功能和卓越..

.NET Framework被淘汰了吗?

.NET Framework并未完全被淘汰,但它的某些版本确实已经停止支持。微软在2023年11月10日停止了对.NET Framework 4.8之前的版本的支持。这一决策意味着使用这些旧版本的应用程序将不再获得安全更新和其他维护,从而可..

强大的 .NET Mock 框架 单元测试模拟库Moq使用教程

单元测试是确保代码质量和可靠性的重要手段。当我们的代码依赖于外部系统、数据库或第三方服务时,编写有效的单元测试可能会变得复杂且耗时。为了简化这一过程,模拟(Mocking)技术应运而生。在 .NET 生态系统中,M..

CLS 问题:超过 0.1(桌面设备) 是什么意思?

在网页设计和开发中,CLS(Cumulative Layout Shift)是指页面布局的累计偏移量。当一个用户与页面互动时,如果页面的某些元素突然改变位置或大小,导致整个布局发生偏移,就会产生布局偏移。这种偏移可能会影响用户..

Angular UT 模拟执行setTimeout

在 Angular 单元测试中,我们经常需要模拟异步操作,比如 setTimeout。提高测试速度: 真实环境下的 setTimeout 会阻塞测试,导致测试运行时间过长。确保测试的可靠性: 模拟 setTimeout 可以让我们更好地控制异步操..

JavaScript 的 sessionStorage 能否加锁?

直接给 sessionStorage 加锁是不可能的。sessionStorage 的本质: sessionStorage 是浏览器提供的一种用于在当前浏览器会话中存储数据的机制。它存储在客户端,数据仅在当前浏览器窗口或标签页中有效。加锁的必要性..

Redis 同步、击穿、穿透及雪崩简述

对Redis最常见的几个问题,简要的说下我的理解与解决方法。数据同步指Redis做为缓存,在数据变化时,怎么保持与数据库数据同步的。一般解决方案为:缓存双删(同步方案大都采用删除缓存,而不会更新新缓存。缓存击穿..

Vue 3.5引入新特性 还有与SSR相关的改进

Vue 3.5引入了响应式属性解构、useTemplateRef方法、useId实用函数、内部响应性重构等新特性。以下是具体介绍:响应式属性解构此功能允许开发者在defineProps宏中解构属性而不会失去响应性,这为组件间的属性传递提..

Asp.Net Core进程内托管 和 进程外托管的区别

在ASP.NET Core中,托管模型决定了应用程序如何运行及其与Web服务器交互的方式。主要有两种托管模式:进程内托管(In-Process Hosting)和进程外托管(Out-of-Process Hosting)。每种模式都有其独特的优势和适用场..

C# 使用Barrier进行多线程同步

在多线程编程中,同步是一个关键问题。Barrier 是 .NET 提供的一种同步机制,用于协调多个线程在执行某个阶段工作时进行等待,直到所有参与的线程都达到某个同步点后再继续执行。这对于需要在多个线程之间进行阶段性..

ASP.NET Core实现多语言本地化Web应用程序

构建全球可访问的网站和应用程序需要对内容进行适当的本地化,以吸引来自不同文化和地区的用户。ASP.NET Core 提供强大的本地化支持,以用户首选的语言和格式呈现内容。在本综合指南中,我们将探索 ASP.NET Core 中..

ASP.NET生成图片验证码

今天开始做一个新项目,包含完整的注册登陆流程,在登陆时需要输入验证码防止暴力破解。 制作思路是这样的: 准备使用handler一般处理程序来写,先随机从0-9和A-Z里随机取4个数字,将内容保存在Session中供验证时使..

C#发送邮件代码简洁示例(附源码下载)

C#发送邮件,主要使用的是System.Net.Mail命名空间下的方法实现,方法很简单,短短十几行代码即可完成发送,具体代码如下。 try { MailMessage myMail = new MailMessage(); myMail.From = new MailAddress..

Docsify | 轻量级无静态构建文档站点生成器

什么是 Docsify?Docsify 可以即时生成你的文档网站。与 GitBook 不同,它不会生成静态 html 文件。相反,它会智能地加载和解析你的 Markdown 文件并将其显示为网站。要开始使用它,你需要做的就是创建一个 index.ht..

发表回复

返回顶部