0

0

c++如何操作定点数运算_c++ 精度丢失预防与数值转换【方法】

冰火之心

冰火之心

发布时间:2026-01-06 19:01:02

|

112人浏览过

|

来源于php中文网

原创

C++无原生定点数类型,需用整数配合统一缩放因子模拟;乘除须显式补偿缩放,加减需同缩放;溢出危险,推荐int64_t并检查范围;转换时应四舍五入,模板封装可提升类型安全。

c++如何操作定点数运算_c++ 精度丢失预防与数值转换【方法】

定点数在 C++ 中没有原生类型,必须手动模拟

C++ 标准库不提供 fixed_point 类型(C++23 才引入实验性 ,但主流编译器尚未完整支持)。所谓“定点数运算”,实际是用整数(int32_tint64_t)存储缩放后的值,再通过固定缩放因子(如 100、65536、1000000)隐式表示小数位。关键不是“用什么类型”,而是“缩放逻辑是否一致”。

  • 缩放因子选 10 的幂(如 1000)便于人眼读写,但除法易引入截断误差;选 2 的幂(如 65536)利于位移优化,且除法可用 >> 替代,但调试时不易直观换算
  • 所有参与运算的变量必须使用同一缩放因子,否则加减会直接错位——比如 a = 12345(代表 12.345,缩放因子 1000),b = 6789(代表 6.789,缩放因子 1000),a + b 才等于 19134 → 19.134;若 b 实际按 100 缩放,结果就完全失真
  • 溢出比浮点更危险:整数溢出是未定义行为(UB),int32_t 存 1000 倍的金额,最大仅支持约 ±2,147,483.647 元;建议优先用 int64_t 并做范围检查

如何安全实现乘除法:缩放因子必须显式参与计算

加减只需同缩放即可;但乘除必须补偿缩放因子,否则结果偏差指数级放大。例如用缩放因子 SCALE = 1000 表示三位小数:

int32_t SCALE = 1000;
int32_t a_fixed = 12345; // 12.345
int32_t b_fixed = 6789;  // 6.789

// ✅ 正确乘法:先乘再降尺度(注意中间可能溢出,需 int64_t 中转) int64_t prod = (int64_t)a_fixed * b_fixed; // = 83814205 int32_t result_mul = (int32_t)(prod / SCALE); // = 83814 → 83.814

// ❌ 错误乘法:直接相乘不缩放 → 83814205 → 解释为 83814.205(错!) // ✅ 正确除法:先升尺度再除(避免过早截断) int64_t div_numerator = (int64_t)a_fixed * SCALE; int32_t result_div = (int32_t)(div_numerator / b_fixed); // ≈ 1819 → 1.819

除法尤其容易漏掉升尺度步骤——直接 a_fixed / b_fixed 会丢掉全部小数位,得到整数商(如 12345 / 6789 = 1),毫无意义。

与浮点数互转时,四舍五入和截断必须明确选择

double 转定点:不加处理直接 static_cast(x * SCALE) 是向零截断(truncation),导致负数偏差更大。金融等场景通常要求四舍五入:

立即学习C++免费学习笔记(深入)”;

Aha
Aha

全天候网红营销AI智能体平台

下载
double x = -12.3456;
int32_t fixed = static_cast(round(x * SCALE)); // → -12346(-12.346)
// 若用 trunc:static_cast(x * SCALE) → -12345(-12.345),误差累积快
  • 从定点转 double 相对简单:static_cast(fixed_val) / SCALE,但要注意 SCALE 必须是 double 类型或强制转换,否则整数除法会截断
  • 批量转换时,避免反复调用 round() 影响性能;可预计算 SCALE_HALF = SCALE / 2,用 (fixed_val >= 0 ? fixed_val + SCALE_HALF : fixed_val - SCALE_HALF) 模拟四舍五入(仅适用于正缩放因子)

std::ratio 和模板封装可提升类型安全

手写 int32_t + 宏定义缩放因子极易出错。C++11 起可用 std::ratio 配合模板构造轻量级定点类,让缩放因子成为类型一部分:

template 
struct fixed {
    T value;
    static constexpr T scale = R::num / R::den;
    fixed(double d) : value(static_cast(std::round(d * scale))) {}
    operator double() const { return static_cast(value) / scale; }
};

using fixed3 = fixed>; fixed3 a{12.345}, b{6.789}; auto c = a.value + b.value; // 编译期确保单位一致

这种写法不能阻止运行时混用不同 fixed<...> 类型(如 fixed>),但至少让缩放逻辑集中、可复用,且构造/转换逻辑收口,减少裸整数误操作。

真正难的不是写几个宏或模板,而是整个项目中所有输入输出、序列化、数据库字段、网络协议字段,都得统一缩放约定——一个 JSON 接口返回 "price": 123.45后端却按 SCALE=100 存进 int32_t,而前端解析时又当 SCALE=1000 用,这种跨层不一致,比精度丢失更致命。

相关专题

更多
json数据格式
json数据格式

JSON是一种轻量级的数据交换格式。本专题为大家带来json数据格式相关文章,帮助大家解决问题。

406

2023.08.07

json是什么
json是什么

JSON是一种轻量级的数据交换格式,具有简洁、易读、跨平台和语言的特点,JSON数据是通过键值对的方式进行组织,其中键是字符串,值可以是字符串、数值、布尔值、数组、对象或者null,在Web开发、数据交换和配置文件等方面得到广泛应用。本专题为大家提供json相关的文章、下载、课程内容,供大家免费下载体验。

531

2023.08.23

jquery怎么操作json
jquery怎么操作json

操作的方法有:1、“$.parseJSON(jsonString)”2、“$.getJSON(url, data, success)”;3、“$.each(obj, callback)”;4、“$.ajax()”。更多jquery怎么操作json的详细内容,可以访问本专题下面的文章。

309

2023.10.13

go语言处理json数据方法
go语言处理json数据方法

本专题整合了go语言中处理json数据方法,阅读专题下面的文章了解更多详细内容。

74

2025.09.10

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

49

2025.08.29

C++中int、float和double的区别
C++中int、float和double的区别

本专题整合了c++中int和double的区别,阅读专题下面的文章了解更多详细内容。

98

2025.10.23

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1006

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

56

2025.10.17

java学习网站推荐汇总
java学习网站推荐汇总

本专题整合了java学习网站相关内容,阅读专题下面的文章了解更多详细内容。

6

2026.01.08

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
WEB前端教程【HTML5+CSS3+JS】
WEB前端教程【HTML5+CSS3+JS】

共101课时 | 8.2万人学习

JS进阶与BootStrap学习
JS进阶与BootStrap学习

共39课时 | 3.1万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号