std::cout

用 std::setprecision 控制小数位数,但必须配合 std::fixed
直接写 std::cout 输出的是 <code>3.1(有效数字2位),不是你想要的“保留2位小数”。这是因为 setprecision 默认控制的是**总有效数字位数**,不是小数点后位数。
要真正让 3.14159 输出为 3.14,得先切到定点表示模式:
std::cout → <code>3.14std::cout → <code>3.14e+00- 一旦用了
std::fixed,后续所有浮点输出都会保持该精度,直到你显式取消或切换模式
不重置流状态会导致后续输出意外失真
很多人在调试时临时加一句 std::cout ,结果发现后面打印的整数也带了小数点,比如 <code>42 变成 42.000 —— 这是因为 std::fixed 和 setprecision 是流的持久状态,不是一次性的格式参数。
安全做法是局部控制:
立即学习“C++免费学习笔记(深入)”;
- 用作用域封装:把格式设置和输出包进一对大括号里,或者写成函数
- 手动恢复:用
std::defaultfloat清除fixed,用std::cout.unsetf(std::ios_base::floatfield)更彻底 - 避免全局污染:别在头文件或初始化代码里无条件设置
fixed
printf 风格更短但类型不安全,std::format(C++20)更推荐
C++20 引入了 std::format,它既安全又简洁:std::format("{:.2f}", 3.14159) 返回 std::string,不会影响流状态,也不依赖 double 实际传入类型。
而传统 printf:printf("%.2f", 3.14159); 看似方便,但有硬伤:
- 编译期不检查参数类型和个数,
printf("%.2f", 42)会 UB(未定义行为) - 不支持自定义类型,无法和
operator 统一管理 - 在 Windows 上默认不启用 C++20 的
std::format,需确认编译器支持(GCC 13+、Clang 15+、MSVC 19.30+)
注意 double 本身精度限制,格式化不能“修复”数值误差
写 std::cout 得到的不是 <code>0.3000000000,而是 0.30000000000000004441。格式化只是控制显示,不改变底层二进制表示。
如果业务要求精确十进制运算(比如金融计算):
- 别依赖
double或float - 改用整数 cents 存储,或引入
boost::multiprecision::cpp_dec_float等库 - 用
std::round(x * 100.0) / 100.0做显示前截断,但仅限展示,不可用于比较或累加
格式化浮点数最麻烦的地方不在语法,而在搞清你要的是“看起来对”,还是“算起来对”。后者永远绕不开表示本质。











