006-nlohmann/json 结构转换-C++开源库108杰

绝大多数情况下,程序和外部交换的数据,都是结构化的数据。

数据战场中的兵种转换

1. 手工实现——必须掌握的基本功

在的业务类型的同一名字空间下,实现 from_json 和 to_json 两个自由函数(必要时,也可定义为类型的友元函数),即可实现该结构类型与 nlohmann/json 数据的双向转换。

示例:

namesapce d2::ec {
struct Order  // 订单
{string id; 	      int customerID;      vector<long> items;  double totalAmount;  string orderDate;  
};// json → Order
void from_json(json const& j, Order& o) 
{j.at("id").get_to(o.id);j.at("customerID").get_to(o.customerID);j.at("items").get_to(o.totalAmount);j.at("totalAmount").get_to(o.totalAmount);j.at("orderDate").get_to(o.orderDate);
}// Order → json
void to_json(json& j, Order const& o)
{j["id"] = o.id;j["customID"] = o.customerID;j["items"] = o.items; // 完美支持 STL 容器j["totalAmount"] = o.totalAmount;j["orderDate"] = o.orderDate;
}} // namespace d2::ec

2. 借助宏,快速定义

  1. NLOHMANN_DEFINE_TYPE_NON_INTRUSIVE // 非侵入式
  2. NLOHMANN_DEFINE_TYPE_INTRUSIVE // 侵入式
  3. NLOHMANN_DEFINE_TYPE_NON_INTRUSIVE_WITH_DEFAULT // 非侵入,且字段缺失时不报错
  4. NLOHMANN_DEFINE_TYPE_INTRUSIVE_WITH_DEFAULT // 侵入式,且字段缺失时不报错
  5. NLOHMANN_DEFINE_DERIVED_TYPE_NON_INTRUSIVE // 用于派生类,非侵入式
  6. NLOHMANN_DEFINE_DERIVED_TYPE_INTRUSIVE // 用于派生类,侵入式
  7. NLOHMANN_DEFINE_DERIVED_TYPE_NON_INTRUSIVE_WITH_DEFAULT //用于派生类,非侵入,字段缺失不报错
  8. NLOHMANN_DEFINE_DERIVED_TYPE_INTRUSIVE_WITH_DEFAULT //用于派生类,侵入,字段缺失不报错
  9. NLOHMANN_JSON_SERIALIZE_ENUM //专用于让枚举类型的值,以字符串方式进行 JSON 读写

3. 视频:快速实现结构转换

011-nlohmann/json-3-结构化转换

4. 代码:我要打十个!

#include <iostream>
#include <string>
#include <vector>
#include <chrono> // 时间
#include <optional> // 可选值#include <nlohmann/json.hpp>using json = nlohmann::ordered_json; namespace d2::ec // d2school 电商系统
{// 第1个:订单状态
enum class OrderStatus // 订单状态
{pending, // 待支付paid, // 已支付    shipped, // 已发货completed, // 已完成cancelled // 已取消
};    NLOHMANN_JSON_SERIALIZE_ENUM(OrderStatus, {{OrderStatus::pending, "pending"},{OrderStatus::paid, "paid"},{OrderStatus::shipped, "shipped"},{OrderStatus::completed, "completed"},{OrderStatus::cancelled, "cancelled"}
})// 第2个:商品
struct Item
{size_t id; // 商品IDstd::string name; // 商品名称double price; // 商品价格double discount = 1; // 商品折扣    
};NLOHMANN_DEFINE_TYPE_NON_INTRUSIVE_WITH_DEFAULT(Item, id, name, price, discount)// 第3个:客户
struct Customer
{std::string id; // 客户IDstd::string nick; // 客户名称
};NLOHMANN_DEFINE_TYPE_NON_INTRUSIVE(Customer, id, nick)// 第4个:是否允许周末送货
enum class WeekendDelivery // 是否允许周末送货
{pending, // 选定allowed, // 允许denied // 拒绝
};NLOHMANN_JSON_SERIALIZE_ENUM(WeekendDelivery, {{WeekendDelivery::pending, "-"},{WeekendDelivery::allowed, "✓"},{WeekendDelivery::denied, "✗"}
})// 第5个:收货地址
struct Address
{std::string name; // 收货人姓名std::string phone; // 收货人电话std::string provinice; // 省std::string city; // 市std::string street; // 街道std::string detail; // 详细地址std::string zip; // 邮政编码WeekendDelivery weekendDelivery = WeekendDelivery::pending; // 是否允许周末送货
};NLOHMANN_DEFINE_TYPE_NON_INTRUSIVE_WITH_DEFAULT(Address, name, phone, provinice, city, street, detail, zip, weekendDelivery)// 第6个:时间点
struct TimePoint : std::chrono::system_clock::time_point
{   using BaseClass = std::chrono::system_clock::time_point;using BaseClass::BaseClass;  // 继承构造函数TimePoint (BaseClass const& tp) : BaseClass(tp) {}     
};void to_json(json& j, TimePoint const& tp)
{auto t = std::chrono::system_clock::to_time_t(static_cast<TimePoint::BaseClass>(tp));char mbstr[100];if (std::strftime(mbstr, sizeof(mbstr), "%Y-%m-%d %H:%M:%S", std::localtime(&t))){j = mbstr; // 转换为字符串}else {j = nullptr; // 转换失败}    
}void from_json(json const& j, TimePoint& tp)
{std::string str = j.get<std::string>();std::tm tm = {};std::istringstream ss(str);ss >> std::get_time(&tm, "%Y-%m-%d %H:%M:%S");if (ss.fail()) {throw std::runtime_error("Failed to parse time point");}std::time_t t = std::mktime(&tm);tp = TimePoint(std::chrono::system_clock::from_time_t(std::mktime(&tm)));
}// 第7个:订单(概要信息)
struct Order // 订单
{std::string id; // 订单ID	      Customer customer; // 客户// 第8个:对 std::vector<> 的先天支持std::vector<Item> items; // 包含商品double totalAmount; // 订单总金额 TimePoint orderTime; // 订单时间Address address; // 收货地址OrderStatus status = OrderStatus::pending; // 订单状态,默认待支付    
};  NLOHMANN_DEFINE_TYPE_NON_INTRUSIVE_WITH_DEFAULT(Order, id, customer, items, totalAmount, orderTime, address, status)// 第9个:订单详情(派生类)
class OrderDetail : private Order // 订单详情
{
public:OrderDetail(Order const& order): Order(order) // 继承构造函数{if (status >= OrderStatus::paid){UpdatePayTime(); // 更新支付时间}}void UpdateMemo(std::string_view m){this->memo = m; // 更新备注}void UpdateStatus(OrderStatus newStatus){if (this->status == newStatus){return; // 状态未改变}if (newStatus == OrderStatus::pending){this->payTime.reset(); // 重置支付时间(变成空)}else if (newStatus >= OrderStatus::paid){if (!this->payTime) // 当前支付时间为空{UpdatePayTime(); // 更新支付时间}}this->status = newStatus; // 更新状态}public:    NLOHMANN_DEFINE_DERIVED_TYPE_INTRUSIVE_WITH_DEFAULT(OrderDetail, Order, memo, payTime)private:    // 更新支付时间void UpdatePayTime(){   this->payTime = TimePoint(std::chrono::system_clock::now());}std::string memo; // 订单备注// 第10个:对 std::optional<> 的先天支持std::optional<TimePoint> payTime; // 支付时间
};}; // namespace d2::ecint main(int, char**)
{using namespace d2::ec;// 创建一个订单Order o1;o1.id = "O-123456"; // 订单IDo1.customer = {"C10026Aed", "南飞的大圣"}; // 客户o1.items =  // 商品{ {1232, "iPhone 14 Pro", 9999.0, 0.8},{452, "MacBook Pro 16", 19999.0, 0.9},{30098, "iPad Pro", 7999.0}};o1.totalAmount = [&item = o1.items] () -> double{double total = 0.0;for (auto const& i : item){                        total += i.price * i.discount;}return total;}();o1.orderTime = TimePoint(std::chrono::system_clock::now()); // 订单时间o1.address = {"孙悟空", "13800138000", "福建省", "厦门市", "沧海路", "天汇大厦908号", "3602001",WeekendDelivery::denied}; // 收货地址o1.status = OrderStatus::pending;// 序列化json j1 = o1; // 序列化为 JSONstd::string jStr = j1.dump(4); // 转换为字符串std::cout << jStr << std::endl; // 打印 JSON// 反序列化json j2 = json::parse(jStr); // 解析 JSONOrder o2 = j2.get<Order>(); // 反序列化为订单对象json j3 = o2; // 序列化为 JSONstd::cout << j3.dump(2) << std::endl; // 打印 JSONstd::cout << "\n=====================================\n";OrderDetail od1(o1); // 创建订单详情json j4 = od1; // 订单详情 -> JSONstd::cout << j4.dump(2) << std::endl; // 打印 JSONstd::cout << "\n------------------------------------------\n";  od1.UpdateStatus(OrderStatus::paid); // 更新状态:已支付od1.UpdateMemo("「商家」:已付款,请尽快发货,走顺风"); // 更新备注json j5 = od1; // 订单详情 -> JSONstd::cout << j5.dump(2) << std::endl; // 打印 JSON
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.pswp.cn/diannao/82977.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

白盒测试——基本路径测试法

一、实验名称 白盒测试——基本路径测试法 二、实验目的 白盒测试是结构测试&#xff0c;是依据被测程序的内部逻辑结构设计测试用例&#xff0c;驱动被测程序运行完成的测试&#xff0c;通过本实验希望&#xff1a; 1、掌握基本路径测试法的基本概念&#xff0c;用具体的例子…

【嵌入模型与向量数据库】

目录 一、什么是向量&#xff1f; 二、为什么需要向量数据库&#xff1f; 三、向量数据库的特点 四、常见的向量数据库产品 FAISS 支持的索引类型 vs 相似度 五、常见向量相似度方法对比 六、应该用哪种 七、向量数据库的核心逻辑 &#x1f50d; 示例任务&#xff1a;…

matlab中和三角函数相关的内容

和三角相关的函数 以下内容为Ai输出 函数描述示例sin(x)正弦函数&#xff0c;返回x的正弦值&#xff0c;x单位为弧度sin(pi/2)cos(x)余弦函数&#xff0c;返回x的余弦值&#xff0c;x单位为弧度cos(pi)tan(x)正切函数&#xff0c;返回x的正切值&#xff0c;x单位为弧度tan(pi…

scratch基础-外观模块

一、本次任务 二、内容详解 1、模块介绍 1、说[你好] (2)秒&#xff1a;临时对话框&#xff0c;短暂对话 2、说[你好]&#xff1a;持续显示对话框&#xff0c;长文本显示 3、思考[嗯…] (2)秒&#xff1a;临时显示思考气泡&#xff0c;用于角色思考 4、思考[嗯…] &#xff1a…

TAOCMS漏洞代码学习及分析

路由规则 分为前台和后台&#xff0c; 前台在api.php中 <?php session_start(); include(config.php); include(SYS_ROOT.INC.common.php); $ctrl$_REQUEST[ctrl]; $action$_REQUEST[action]; $mucfirst($action); if(!in_array($m,array(Api,Comment)))d…

Spring @Scheduled注解详解

文章目录 1.Scheduled注解定义2.配置 Scheduled2.1 开启定时任务支持2.2 创建定时任务 3. 常用属性3.1 fixedRate3.2 fixedDelay3.3 cron 4.工作原理4.1 基于TaskScheduler4.2 使用 ThreadPoolTaskScheduler4.3 定时任务的执行流程 5. 延时执行的定时任务5.1 创建定时任务类5.2…

理解计算机系统_并发编程(5)_基于线程的并发(二):线程api和基于线程的并发服务器

前言 以<深入理解计算机系统>(以下称“本书”)内容为基础&#xff0c;对程序的整个过程进行梳理。本书内容对整个计算机系统做了系统性导引,每部分内容都是单独的一门课.学习深度根据自己需要来定 引入 接续上一篇理解计算机系统_并发编程(4)_基于线程的并发(一…

使用PhpStudy搭建Web测试服务器

一、安装PhpStudy 从以下目录下载PhpStudy安装文件 Windows版phpstudy下载 - 小皮面板(phpstudy) (xp.cn) 安装成功之后打开如下界面 点击启动Apache 查看网站地址 在浏览器中输入localhost:88,出现如下页面就ok了 二、与Unity交互 1.配置下载文件路径&#xff0c;点击…

cocos creator 3.8 下的 2D 改动

在B站找到的系统性cocos视频教程,纯2D开发入门,链接如下: zzehz黑马程序员6天实战游戏开发微信小程序&#xff08;Cocos2d的升级版 CocosCreator JavaScript&#xff09;_哔哩哔哩_bilibili黑马程序员6天实战游戏开发微信小程序&#xff08;Cocos2d的升级版 CocosCreator Ja…

【Hot 100】208. 实现 Trie (前缀树)

目录 引言实现 Trie (前缀树)我的解题代码解析代码思路分析优化建议1. 内存泄漏问题2. 使用智能指针优化内存管理3. 输入合法性校验&#xff08;可选&#xff09;4. 其他优化 总结 &#x1f64b;‍♂️ 作者&#xff1a;海码007&#x1f4dc; 专栏&#xff1a;算法专栏&#x1…

Unity3D仿星露谷物语开发42之粒子系统

1、目标 使用例子系统&#xff0c;实现割草后草掉落的特效。 通过PoolManager获取特效预制体&#xff0c;通过VFXManager来触发特效。 2、配置例子特效 在Hierarchy -> PersistentScene下创建新物体命名为Reaping。 给该物体添加Particle System组件。 配置例子系统参数…

视觉-语言基础模型作为高效的机器人模仿学习范式

摘要 近期&#xff0c;视觉语言基础模型领域取得的进展彰显了其在理解多模态数据以及解决复杂视觉语言任务&#xff08;包括机器人操作任务&#xff09;方面的能力。我们致力于探寻一种简便的方法&#xff0c;利用现有的视觉语言模型&#xff08;VLMs&#xff09;&#xff0c;仅…

zst-2001 上午题-历年真题 算法(5个内容)

回溯 算法 - 第1题 找合适的位置&#xff0c;如果没有位置就按B回家 d 分治 算法 - 第2题 b 算法 - 第3题 a 算法 - 第4题 划分一般就是分治 a 算法 - 第5题 分治 a 0-1背包 算法 - 第6题 c 算法 - 第7题 最小的为c 3100 c 算法 - 第8题 …

浅论3DGS溅射模型在VR眼镜上的应用

摆烂仙君小课堂开课了&#xff0c;本期将介绍如何手搓VR眼镜&#xff0c;并将随手拍的电影变成3D视频。 一、3DGS模型介绍 3D 高斯模型是基于高斯函数构建的用于描述三维空间中数据分布概率的模型&#xff0c;高斯函数在数学和物理领域有着广泛应用&#xff0c;其在 3D 情境下…

2025年中期大语言模型实力深度剖析

I. 引言&#xff1a;解读2025年动态LLM竞技场中的“实力” 用户提出的“如今哪个大语言模型最强”这一问题&#xff0c;精准地反映了业界对飞速发展的人工智能&#xff08;AI&#xff09;领域的高度关注。本报告基于截至2025年5月的最新数据&#xff0c;旨在对这一问题进行全面…

Spark缓存-cache

一、RDD持久化 1.什么时候该使用持久化&#xff08;缓存&#xff09; 2. RDD cache & persist 缓存 3. RDD CheckPoint 检查点 4. cache & persist & checkpoint 的特点和区别 特点 区别 二、cache & persist 的持久化级别及策略选择 Spark的几种持久化…

嵌入式开发学习日志(数据结构--顺序结构单链表)Day19

一、顺序结构 安装软件命令&#xff1a; sudo apt-get install (软件名) 安装格式化对齐&#xff1a;sudo apt-get install clang-format 内存泄漏检测工具&#xff1a; sudo apt-get install valgrind 编译后&#xff0c;使用命令 valgrind ./a.out 即可看内…

第六节第二部分:抽象类的应用-模板方法设计模式

模板方法设计模式的写法 建议使用final关键字修饰模板方法 总结 代码&#xff1a; People(父类抽象类) package com.Abstract3; public abstract class People {/*设计模板方法设计模式* 1.定义一个模板方法出来*/public final void write(){System.out.println("\t\t\t…

2025年渗透测试面试题总结-渗透测试红队面试三(题目+回答)

网络安全领域各种资源&#xff0c;学习文档&#xff0c;以及工具分享、前沿信息分享、POC、EXP分享。不定期分享各种好玩的项目及好用的工具&#xff0c;欢迎关注。 目录 渗透测试红队面试三 六十一、主机被入侵自查解决方案 六十二、NAT&#xff08;网络地址转换&#xff…

springboot-web基础

21.web spring MVC 基于浏览器的 B/S 结构应用十分流行。Spring Boot 非常适合 Web 应用开发。可以使用嵌入式 Tomcat、Jetty、 Undertow 或 Netty 创建一个自包含的 HTTP 服务器。一个 Spring Boot 的 Web 应用能够自己独立运行&#xff0c;不依赖需 要安装的 Tomcat&#x…