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

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.xdnf.cn/news/6077.html

相关文章:

  • # 深度剖析LLM的“大脑”:单层Transformer的思考模式探索
  • 青少年编程与数学 02-019 Rust 编程基础 11课题、类型系统
  • GAN简读
  • npm install 报错
  • CS4334:一款高性能的立体声音频数模转换器
  • 如何自定义 Spring MVC 的配置?
  • 【unity游戏开发——编辑器扩展】使用EditorGUI的EditorGUILayout绘制工具类在自定义编辑器窗口绘制各种UI控件
  • 高速数字测试利器,新款是德科技UXR0504B示波器
  • RPA vs. 传统浏览器自动化:效率与灵活性的终极较量
  • STM32 片上资源之串口
  • 《实现模式》以Golang视角解读 价值观和原则 day 1
  • 快速定位到源码位置的插件 - vite/webpack
  • 【Python】普通方法、类方法和静态方法的区分
  • hbase shell的常用命令
  • 双目云台摄像机:双摄安防功能全方位
  • Java运行原理分析
  • LeetCode 热题 100 114. 二叉树展开为链表
  • Spring的bean的生命周期?
  • 【机器学习】支持向量回归(SVR)从入门到实战:原理、实现与优化指南
  • 各大编程语言基本语法区别
  • 游戏引擎学习第279天:将实体存储移入世界区块
  • 为什么 Linux 上默认没有 host.docker.internal
  • 【内网渗透】——NTML以及Hash Relay
  • MySQL Explain 中 Type 与 Extra 字段详解
  • MySQL 服务器配置和管理(上)
  • 监听用户切换浏览器标签页,切换回页面刷新页面
  • 代码随想录算法训练营第60期第三十五天打卡
  • 嵌入式自学第二十天(5.13)
  • AIStarter新功能上线:模型管理与创作者收益系统全面升级,助力AI开发效率提升
  • 函数定义、 异常处理、 迭代器协议、内置函数、返回值