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

JSON 快速上手:语法解析与应用实例

个人主页:chian-ocean

文章专栏-Linux

前言:

JSON(JavaScript Object Notation)是一种轻量级的数据交换格式,易于人类阅读和编写,也易于机器解析和生成。它常用于客户端和服务器之间的数据传输,尤其是在 Web 开发中非常常见。

在这里插入图片描述

JSON库的安装

yum安装

sudo yum install jsoncpp-devel

image-20250611125828145

  • 这里面是已经安装好的Nothing to do

JSON介绍

JSON(JavaScript 对象表示法)是一种轻量级的数据交换格式,设计上简洁易读。它是基于文本的,易于人类理解和编写,也易于机器解析和生成。JSON 经常用于 Web 应用中,尤其是前端与后端之间的数据交换。

JSON的基本结构

SON 主要有两种数据结构:

  • 对象(Object):由一组由键(键值对)组成的无序集合。对象在 JSON 中由大括号 {} 包围。
{"name": "Alice","age": 25,"isStudent": true
}
  • 数组(Array):由多个按顺序排列的值组成。数组在 JSON 中由方括号 [] 包围。
["apple", "banana", "cherry"]
  • 嵌套对象和数组:JSON 允许对象和数组嵌套,即一个 JSON 对象的值可以是另一个对象或数组。
{"name": "John","age": 30,"isStudent": false,"courses": ["Math", "Physics", "Chemistry"],"address": {"street": "123 Main St","city": "New York","zip": "10001"},"spouse": null
}

JSON的数据类型

  • 字符串(String):必须用双引号 " 括起来,支持 Unicode 字符。
  • 数字(Number):可以是整数或浮动数,且没有引号。
  • 布尔值(Boolean):可以是 truefalse
  • 数组(Array):由零个或多个数据项组成,数据项之间用逗号分隔。
  • 对象(Object):由一对一对的键值对组成,键和值之间用冒号 : 分隔,多个键值对之间用逗号 , 分隔。
  • null:表示一个空值

Jsoncpp使用

JSON对象的创建

[[include]] <iostream>
[[include]] <string>
[[include]] <jsoncpp/json/json.h>
using namespace std;int main() {// 创建一个 JSON 对象Json::Value root;// 向 JSON 对象中添加键值对root["name"] = "Alice";        // 字符串root["age"] = 25;              // 数字root["isStudent"] = true;      // 布尔值cout << root.toStyledString() <<endl;   //打印return 0;
}

输出

image-20250611125847311

FastWriter转换成字符串风格

  • FastWriterjsoncpp 中的一个类,它用于将 Json::Value 对象转换为 JSON 字符串,且具有高效的序列化速度
  • FastWriter 会生成一个紧凑的 JSON 字符串(没有缩进和换行),非常适合于数据传输和存储。
  • FastWriterStreamWriter 类相比,速度较快,但它不支持格式化输出(即不支持缩进和换行),所以适用于需要高效、简洁的输出场景。
[[include]] <iostream>
[[include]] <string>
[[include]] <jsoncpp/json/json.h>  // 引入 jsoncpp 库头文件
using namespace std;int main() {// 创建一个 JSON 对象Json::Value root;// 向 JSON 对象中添加键值对root["name"] = "Alice";        // 添加字符串键值对,"name" 键对应 "Alice"root["age"] = 25;              // 添加数字键值对,"age" 键对应 25root["isStudent"] = true;      // 添加布尔值键值对,"isStudent" 键对应 true// 输出格式化的 JSON 字符串,带有缩进和换行cout << "Formatted JSON Output:" << endl;cout << root.toStyledString() << endl;// 使用 FastWriter 生成紧凑的 JSON 字符串(没有缩进和换行)Json::FastWriter w;  // 创建 FastWriter 对象string jsonstr = w.write(root);  // 将 JSON 对象转换为紧凑的 JSON 字符串// 输出紧凑格式的 JSON 字符串cout << "Compact JSON Output (FastWriter):" << endl;cout << "jsonstr: " << jsonstr << endl;return 0; 
}

输出

image-20250611125851592

Reader转换成value

jsoncpp 中,如果你想将 JSON 字符串(std==string 类型)转换为 Json==Value 对象,可以使用 Json==ReaderJson==CharReader 来进行解析。Json==Valuejsoncpp 中用来表示 JSON 数据的核心类,Json==Reader 用于将 JSON 字符串解析为 Json::Value 对象。

[[include]] <iostream>
[[include]] <string>
[[include]] <jsoncpp/json/json.h>  // 引入 jsoncpp 库头文件
using namespace std;int main() {// 创建一个空的 JSON 对象Json::Value root;// 向 JSON 对象中添加键值对root["name"] = "Alice";        // 字符串:向 root 中添加键为 "name" 的字符串 "Alice"root["age"] = 25;              // 数字:向 root 中添加键为 "age" 的整数 25root["isStudent"] = true;      // 布尔值:向 root 中添加键为 "isStudent" 的布尔值 true// 输出格式化的 JSON 对象(带缩进和换行)cout <<"root:   \n" << root.toStyledString() << endl;// toStyledString() 会输出格式化的 JSON 字符串(带缩进和换行,易于阅读)// 使用 FastWriter 生成紧凑格式的 JSON 字符串(没有缩进和换行)Json::FastWriter w;  // 创建 FastWriter 对象 wstring jsonstr = w.write(root);  // 将 root 转换为紧凑格式的 JSON 字符串,并存储在 jsonstr 中// 输出紧凑格式的 JSON 字符串cout << "jsonstr: " << jsonstr << endl;// 由于 FastWriter 会去掉换行和空格,这里的输出不会有任何格式化// 创建 Json::Reader 对象,用于解析 JSON 字符串Json::Reader r;// 创建一个 Json::Value 对象 res 用于存储解析后的 JSON 数据Json::Value res;// 使用 Json::Reader 对象解析 JSON 字符串 jsonstrif (r.parse(jsonstr, res)) {// 如果解析成功,输出格式化后的 JSON 对象cout <<"res:  \n" << res.toStyledString() << endl;} else {// 如果解析失败,输出错误信息cout << "Failed to parse the JSON string!" << endl;}return 0; 
}

输出:

image-20250611125855337

将序列化与反序列化用JSON改写

Requset

序列化

bool Serialize(std::string *out)
{
[[ifdef]] Myself// 构建有效载荷// 将x_转换为字符串并加入空格分隔符std==string s = std==to_string(x_);s += space_sep;  // 添加空格分隔符s += op_;        // 添加操作符s += space_sep;  // 添加空格分隔符s += std::to_string(y_);  // 将y_转换为字符串并添加*out = s;  // 将构建的字符串赋值给输出参数return true;
[[else]]// 如果不是在Myself模式下,使用JSON格式序列化Json::Value root;root["x"] = x_;  // 将x_值放入JSON根节点root["y"] = y_;  // 将y_值放入JSON根节点root["op"] = op_;  // 将操作符op_放入JSON根节点Json::FastWriter w;  // 创建一个JSON序列化工具*out = w.write(root);  // 使用JSON工具将根节点序列化为字符串并赋值给输出参数return true;
[[endif]]      
}

Myself宏定义的情况下,序列化数据为一个简单的字符串格式,通过空格分隔x_op_y_

如果Myself宏没有定义,则使用Json::Value将数据序列化为JSON格式。

反序列化

// 反序列化 x + y
bool Deserialize(const std::string &in)
{
[[ifdef]] Myself// 解析有效载荷// 查找第一个空格分隔符的位置auto pos1 = in.find(space_sep);if (pos1 == std==string==npos)  // 如果没有找到空格分隔符,返回失败return false;// 提取x的部分std::string part_x = in.substr(0, pos1);// 查找最后一个空格分隔符的位置auto pos2 = in.rfind(space_sep);// 提取操作符部分std::string oper = in.substr(pos1 + 1, pos2);// 提取y的部分std::string part_y = in.substr(pos2 + 1);// 如果空格分隔符之间的距离不为2,说明格式不正确if (pos2 != pos1 + 2)return false;// 解析操作符(目前假设是单字符)op_ = in[pos1 + 1];// 将x和y从字符串转换为整数x_ = std::stoi(part_x);y_ = std::stoi(part_y);return true;
[[else]]// 如果不是在Myself模式下,使用JSON格式反序列化Json::Value root;Json::Reader r;// 解析输入字符串为JSON格式r.parse(in, root);// 提取x、y和op的值x_ = root["x"].asInt();  // 提取x的整数值y_ = root["y"].asInt();  // 提取y的整数值op_ = root["op"].asString();  // 提取操作符的字符串值return true;
[[endif]]
}

Response

序列化

// 序列化 r code
bool Serialize(std::string *out)
{
[[ifdef]] Myself// 构建有效载荷// 将result_转换为字符串std==string s = std==to_string(result_);// 添加空格分隔符s += space_sep;// 将code_转换为字符串并添加s += std::to_string(code_);// 将构建的字符串赋值给输出参数*out = s;return true;[[else]]// 如果不是在Myself模式下,使用JSON格式序列化Json::Value root;// 将result_和code_添加到JSON对象中root["result"] = result_;root["code"] = code_;Json::FastWriter w;  // 创建一个JSON序列化工具// 使用JSON工具将数据序列化为字符串并赋值给输出参数*out = w.write(root);return true;
[[endif]]
}

反序列化

bool Deserialize(const std::string &in)
{[[ifdef]] Myself// 查找空格分隔符的位置auto pos = in.find(space_sep);// 提取result_的部分(空格之前的部分)std::string res = in.substr(0, pos);// 提取code_的部分(空格之后的部分)std::string code = in.substr(pos + 1);// 如果空格分隔符的位置不唯一,说明格式不正确if (pos != in.rfind(space_sep))return false;// 将提取的result_和code_从字符串转换为整数result_ = std::stoi(res);code_ = std::stoi(code);return true;[[else]]// 如果不是在Myself模式下,使用JSON格式反序列化Json::Value root;Json::Reader r;// 解析输入的字符串为JSON格式r.parse(in, root);// 提取JSON中的result和code值,未赋值到result_和code_,需要修正为赋值操作result_ = root["result"].asInt();  // 提取result并赋值给result_code_ = root["code"].asInt();  // 提取code并赋值给code_return true;[[endif]]
}
http://www.dtcms.com/a/354904.html

相关文章:

  • 【VSCode】使用VSCode打开md文件以及转化为PDF
  • 打工人日报#20250828
  • HTTP 分块传输编码:深度解析与报文精髓
  • 第21节:环境贴图与PBR材质升级——构建电影级真实感渲染
  • Java 实现HTML转Word:从HTML文件与字符串到可编辑Word文档
  • 腕上智慧健康管家:华为WATCH 5与小艺的智美生活新范式
  • 使用EasyExcel实现Excel单元格保护:自由锁定表头和数据行
  • mac电脑双屏显示时程序坞跑到副屏的解决方法
  • 吱吱企业通讯软件以安全为基,搭建高效的通讯办公平台
  • ckman部署的clickhouse,节点迁移
  • 微算法科技(NASDAQ:MLGO)推出创新型混合区块链共识算法,助力物联网多接入边缘计算
  • [论文阅读] 人工智能 + 软件工程 | 告别“隐藏陷阱”:领域预训练模型SmartBERT如何赋能智能合约安全
  • MyBatis题
  • AR培训系统:油气行业的安全与效率革新
  • List<Map<String, String>>最简单的遍历方式
  • 在Ubuntu中安装配置MySql Server
  • [光学原理与应用-320]:光学产品不同阶段使用的工具软件、对应的输出文件
  • 计算机考研408《数据结构》真题模拟——数据结构与算法基本概念
  • DQN(深度Q网络):深度强化学习的里程碑式突破
  • Java 线程池拒绝策略
  • vscode pyqt5设置
  • 基于SpringBoot的老年人健康数据远程监控管理系统【2026最新】
  • JavaSE——八股文
  • 医院信息系统(HIS)的开发架构解析,代码示例
  • 面试tips--并发--进程与线程的区别线程通信方式总结
  • k8s集群1.20.9
  • 虚拟相机的最佳实践参考是什么
  • k8s是什么?
  • docker和k8s的区别
  • Android 开发 - 数据共享(数据共享、内容提供者实现、动态权限申请)