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

解决 CJSON 浮点数精度问题:从 `cJSON_AddNumberToObject` 到 `cJSON_AddRawToObject`

在使用 CJSON 库处理浮点数时,开发者常会遇到一个棘手问题:浮点数的小数位精度丢失。例如,数值 3.1400 可能被简化为 3.14,甚至 5.0 被显示为 5。这种默认行为在需要严格保留小数位的场景(如金融、物联网传感数据)中会引发严重问题。

本文将分析默认方法 cJSON_AddNumberToObject 的局限性,并介绍如何通过 cJSON_AddRawToObject 实现浮点数精度的精确控制。


一、问题分析:cJSON_AddNumberToObject 的局限性

  1. 默认行为
    cJSON_AddNumberToObject 是 CJSON 提供的快捷函数,用于向 JSON 对象中添加数值。其内部实现依赖 snprintf%g 格式符,自动省略无效的小数位和末尾零。例如:
cJSON_AddNumberToObject(root, "temperature", 23.50);
cJSON_AddNumberToObject(root, "pi", 3.140000);

序列化结果可能为:

{"temperature": 23.5,  // 末尾零被删除"pi": 3.14            // 多余小数位被截断
}
  1. 问题根源
    • 格式符 %g 的自动优化:%g 会删除无效零并缩短浮点表示(如 3.14003.14)。

• 数据精度丢失:若接收方依赖固定小数位数(如金额必须两位小数),默认行为会导致数据错误。


二、解决方案:cJSON_AddRawToObject 的精准控制

  1. 核心思路
    通过 cJSON_AddRawToObject 直接传递格式化后的数值字符串,绕过 CJSON 的默认序列化逻辑,从而完全控制小数位。

  2. 关键函数
    cJSON_AddRawToObject

向 JSON 对象中添加键值对,值的部分直接使用原始字符串(需符合 JSON 数值格式)。

void cJSON_AddRawToObject(cJSON *object, const char *name, const char *raw);

object: 目标 JSON 对象。

name: 键名。

raw: 值的字符串表示(如 "3.14")。

  1. 代码示例
#include <stdio.h>
#include "cJSON.h"int main() {cJSON* root = cJSON_CreateObject();double temperature = 23.5;double pi = 3.1415926;// 1. 默认方法(精度丢失)cJSON_AddNumberToObject(root, "temperature_bad", temperature);cJSON_AddNumberToObject(root, "pi_bad", pi);// 2. 精准控制方法char temp_buffer[32], pi_buffer[32];snprintf(temp_buffer, sizeof(temp_buffer), "%.2f", temperature); // 格式化为两位小数snprintf(pi_buffer, sizeof(pi_buffer), "%.4f", pi);               // 格式化为四位小数cJSON_AddRawToObject(root, "temperature_good", temp_buffer);cJSON_AddRawToObject(root, "pi_good", pi_buffer);// 输出结果char* json_str = cJSON_Print(root);printf("%s\n", json_str);cJSON_Delete(root);free(json_str);return 0;
}
  1. 输出结果
{"temperature_bad": 23.5,      // 默认方法:精度丢失"pi_bad": 3.1415926,          // 实际可能显示为 3.141593(精度问题)"temperature_good": 23.50,    // 强制两位小数"pi_good": 3.1416             // 强制四位小数(自动四舍五入)
}

三、关键注意事项

  1. 数据类型保证
    cJSON_AddRawToObject 生成的是 number 类型,而非字符串。

正确示例:"3.14" → 解析为数值。
错误示例:"3.14元" → 非数值格式,解析失败。

  1. 缓冲区溢出防护
    • 使用 snprintf 限制写入长度,避免内存越界:
char buffer[32];
snprintf(buffer, sizeof(buffer), "%.2f", value); // 安全写法
  1. 四舍五入处理
    snprintf 的格式化会自动四舍五入:
double value = 3.1415926;
snprintf(buffer, sizeof(buffer), "%.2f", value); // 输出 "3.14"
snprintf(buffer, sizeof(buffer), "%.3f", value); // 输出 "3.142"
  1. 性能影响
    • 优点:直接控制精度,无需后处理。

• 缺点:频繁格式化字符串可能增加 CPU 开销。


四、扩展场景:科学计数法与大数据

  1. 科学计数法
    若数值过大或过小,可强制使用科学计数法:
double value = 0.00000314;
snprintf(buffer, sizeof(buffer), "%.2e", value); // 输出 "3.14e-06"
  1. 动态精度控制
    在业务逻辑中动态调整小数位数:
int precision = 2;
snprintf(buffer, sizeof(buffer), "%.*f", precision, value); // 保留两位小数

五、总结

  1. 方法对比
    | 方法 | 优点 | 缺点 |
    |-------------------------|-------------------------------|-----------------------|
    | cJSON_AddNumberToObject | 简单快捷 | 无法控制精度 |
    | cJSON_AddRawToObject | 精准控制小数位,保留数值类型 | 需手动格式化字符串 |

  2. 推荐场景
    • 严格精度需求(如金融、传感器数据):使用 cJSON_AddRawToObject

• 临时调试或非关键数据:使用默认 cJSON_AddNumberToObject

  1. 终极建议
    在关键业务逻辑中,永远不要依赖默认浮点序列化。通过 cJSON_AddRawToObject 显式控制精度,可避免因数据格式问题引发的系统性风险。

相关文章:

  • 大项目k8s集群有多大规模,多少节点,有多少pod
  • 基于 Cursor + 浏览器MCP服务 实现 Web端自动化测试
  • 【Dv3Admin】工具数据验证配置文件解析
  • Python-Flask-Dive
  • mapbox进阶,使用mapbox-plugins插件加载饼状图
  • 【Python】Python常用数据类型详解
  • 一周学完计算机网络之三:1、数据链路层概述
  • 安装Hadoop并运行WordCount程序
  • ACL访问控制列表:access-list 10 permit 192.168.10.1
  • MySQL-逻辑架构
  • (五)毛子整洁架构(分布式日志/Redis缓存/OutBox Pattern)
  • 不定长滑动窗口---初阶篇
  • 科技快讯 | 字节跳动开源一款Deep Research项目;全球首个氮化镓量子光源芯片发布
  • 图论拓扑排序
  • Spring Boot动态配置修改全攻略
  • 神经符号推理系统:研究进展与应用前景
  • MCP 入门实战:用 C# 开启 AI 新篇章
  • 文本数据可视化
  • 【软件工程】软件缺陷 基于组合的优化方法
  • Python爬虫实战:研究ajax异步渲染加密
  • 俄总统新闻秘书:普京提议谈判表明俄寻求和平解决方案意愿
  • 俄乌释放停火和谈信号,克宫:将组建“相应级别”谈判代表团
  • 数说母亲节|全球11亿女性进入更年期,“不是忍忍就好”
  • 富家罹盗与财富迷思:《西游记》与《蜃楼志》中的强盗案
  • 苏东坡:人生就是哈哈哈哈哈
  • 时代中国控股:前4个月销售额18.1亿元,境外债重组协议押后聆讯至5月底