初识c++;format函数;cout输出

一.程序

在c语言中输入和输出的头文件是#include <stdio.h>,而在c++中标准输入输出头文件则是#include <iostream>,它的作用和stdio类似,不过更为强大

#include <iostream> int main(){ std::cout<<"Hello World!"<<std::endl; }

1.cout   这是iostream文件里的内容

2.std::  c++中有个命名空间的概念,在这里std::cout的作用是在告诉程序我们在使用标准库里的cout

3.cout  代表标准输出流对象,作用是将数据以字符的形式打印到终端

4.<<  是cout的特殊操作符,代表右侧的内容流向cout

5.endl  与\n是一个意思,唯一的差别是在某些情况下,输出的内容可能并不一定会立即显示在终端上,而endl可以确保立即显示

简便写法:

#include <iostream> using namespace std; int main(){ cout<<"Hello World!"<<endl; }

1.using namespace std;  代表这段程序使用这个命名空间的内容时,不再需要加前缀,这样不用每次都加std::

2.namespace 代表命名空间,用于解决重名冲突

3.using namespace  代表这段程序默认使用某个命名空间

4.std  代表标准库的命名空间

二.输入与输出

#include <iostream> using namespace std; int mian(){ int a; cin>>a; cout<<a<<endl; }

1.cin  和cout一样都是在头文件iostream中定义的标准输入对象,作用是将终端中的字符出入给程序的变量

三.format函数

1、std::format 核心介绍

std::format 是 C++20 引入的类型安全的字符串格式化工具,核心定位是 “统一且优雅的格式化解决方案”,定义在<format>头文件中,核心能力包括:

  1. 格式化字符串生成:将变量按指定格式拼接成字符串(而非直接输出),再配合cout/ 文件等输出;
  2. 语法灵活:支持位置占位符、精度控制、对齐、进制转换等;
  3. 类型安全:编译期检查类型匹配,避免格式符与变量类型不匹配的运行时错误;
  4. 扩展性强:可自定义类 / 结构体的格式化规则。

基础示例(生成格式化字符串后输出):

#include <iostream> #include <format> using namespace std; int main() { int age = 20; double score = 95.567; // 生成格式化字符串 string info = format("年龄:{:>3d},成绩:{:.2f}", age, score); // 配合cout输出 cout << info << endl; // 输出:年龄: 20,成绩:95.57 return 0; }

2. 核心语法规则(必掌握)

语法格式作用说明示例代码输出结果
{n}指定变量位置(n 从 0 开始)format("{1} {0}", "B", "A")A B
{:.2f}浮点数保留 2 位小数format("{:.2f}", 3.1415)3.14
{:d}十进制整数(默认)format("{:d}", 123)123
{:x}/{:X}十六进制(小写 / 大写)format("{:X}", 255)FF
{:05d}整数补零到 5 位format("{:05d}", 123)00123
{:>8s}字符串右对齐,总宽度 8format("{:>8s}", "abc")" abc"
{:<8s}字符串左对齐,总宽度 8format("{:<8s}", "abc")"abc "
{:+d}显示整数符号(+/-)format("{:+d}", -123)-123
{:.3e}科学计数法,保留 3 位有效数字format("{:.3e}", 1234.56)1.235e+03

实战示例(组合规则)

#include <iostream> #include <format> using namespace std; int main() { int age = 25; double salary = 12345.6789; int id = 78; // 组合格式化规则:位置+补零+小数精度 string info = format( "员工ID:{2:04d} | 年龄:{0:d} | 薪资:{1:+.2f}", age, salary, id ); cout << info << endl; // 输出:员工ID:0078 | 年龄:25 | 薪资:+12345.68 return 0; }

3.三类输入输出方式对比

特性scanf/printf (C 风格)cin/cout (C++ 流)std::format + cin/cout (C++20)
类型安全❌ 不安全(编译期不检查)✅ 安全(编译期类型检查)✅ 安全(编译期严格检查)
语法直观性❌ 格式符繁琐(% d/% f/% s)✅ 语法简洁(>>/<<)✅ 占位符清晰({})
格式化能力✅ 强(但格式符易出错)❌ 弱(格式化需额外操作)✅ 极强(灵活且易读)
性能✅ 快(接近底层)❌ 慢(默认同步 C 流)⚠️ 中等(略低于 printf,远高于 cout)
扩展性❌ 差(不支持自定义类型)✅ 强(重载 >>/<< 即可)✅ 极强(重载 formatter)
兼容性✅ 全兼容(所有 C/C++ 版本)✅ 兼容 C++98+❌ 仅 C++20+(低版本需 fmtlib)
错误处理❌ 隐式错误(返回值易忽略)✅ 可通过流状态检查错误✅ 抛出 format_error 异常
1. scanf/printf(C 风格)

核心优势

  • 性能极高:直接操作底层缓冲区,无 C++ 流的封装开销,适合高性能场景(如大量数据读写);
  • 兼容性无敌:所有 C/C++ 编译器都支持,无版本限制;
  • 格式化紧凑:一行代码可完成复杂格式化(如printf("%.2f %X", 3.14, 255))。

核心痛点

  • 类型不安全:比如用%d接收double类型,编译不报错,运行时崩溃 / 输出乱码;
  • 易出错:格式符与变量数量 / 类型不匹配、缓冲区溢出(scanf 无边界检查);
  • 不支持自定义类型:无法直接格式化自定义类(如Point{1,2})。
2. cin/cout(C++ 流)

核心优势

  • 类型安全:编译期检查类型,比如cin >> int_var只能接收整数,否则流状态置错;
  • 语法自然:cin >> a >> bcout << "age: " << age,无需记忆格式符;
  • 支持自定义类型:重载operator>>/operator<<即可格式化自定义类。

核心痛点

  • 性能差:默认同步 C 标准流(stdio),导致速度比 printf 慢 1-2 个数量级(可通过ios::sync_with_stdio(false); cin.tie(nullptr);优化,但仍不如 printf);
  • 格式化繁琐:比如保留 2 位小数需要cout << fixed << setprecision(2) << score,代码冗长;
  • 无占位符灵活度:无法直接调整输出位置、进制(需额外操作)。
3. std::format + cin/cout(C++20)

核心优势(对比前两者的核心提升):

  • 兼顾类型安全与格式化能力:既像 cin/cout 一样编译期检查类型,又像 printf 一样支持复杂格式化,且语法更清晰;
  • 格式化语法更优雅:
    • printf:printf("姓名:%s,年龄:%d,成绩:%.2f", name, age, score)
    • format:format("姓名:{},年龄:{},成绩:{:.2f}", name, age, score)
    • 无需记忆格式符,占位符与变量一一对应,位置可自由调整(如{1} {0});
  • 生成字符串而非直接输出:可先格式化字符串(如string s = format(...)),再按需输出到控制台 / 文件 / 网络,灵活性远超 printf/cout;
  • 扩展性更强:自定义类型格式化更简洁(重载formatter),且支持更多格式化场景(如容器、日期)。

核心痛点

  • 仅 C++20 + 支持:低版本编译器(如 GCC 9、MSVC 2019 之前)无法直接使用,需依赖第三方库fmtlibformat的前身,语法完全兼容);
  • 性能略低于 printf:但远高于未优化的 cin/cout,且日常开发中性能差异可忽略。

四.cout

一、cout 核心介绍

cout 是 C++ 标准库<iostream>中定义的标准输出流对象,全称为std::cout,核心作用是将数据输出到控制台(终端),是 C++ 替代 C 语言printf的面向对象输出方式,核心特点:

  • 类型安全:编译期检查输出数据的类型,避免printf的格式符匹配错误;
  • 语法简洁:使用<<(插入运算符)串联输出内容,无需记忆格式符;
  • 可扩展:支持自定义类型的输出(重载<<运算符);
  • 默认性能较低:因同步 C 语言的stdio流,可手动优化。

二、cout 基础使用教程

1. 环境准备

只需包含核心头文件<iostream>,新手可直接使用using namespace std;简化代码:

#include <iostream> // 必须包含的头文件 // 可选:避免每次写std::cout using namespace std;
2.核心语法说明
语法 / 关键字作用说明示例代码输出结果
cout <<插入运算符,输出数据cout << 123;123
endl输出换行符并刷新缓冲区cout << "test" << endl;test(换行)
\n仅输出换行符(不刷新缓冲区,更快)cout << "test\n";test(换行)
boolalpha布尔值输出 true/false(而非 1/0)cout << boolalpha << false;false
noboolalpha恢复布尔值数字输出cout << noboolalpha << true;1
flush手动刷新输出缓冲区cout << "test" << flush;test(无换行)

三、cout 格式化输出(重点)

cout 本身格式化能力较弱,需配合<iomanip>头文件中的格式化操纵符,实现精度、对齐、进制等控制:

#include <iostream> #include <iomanip> // 必须包含,提供格式化操纵符 using namespace std; int main() { double pi = 3.1415926; int num = 255; string str = "abc"; // 1. 浮点数精度控制(保留2位小数) cout << "π保留2位小数:" << fixed << setprecision(2) << pi << endl; // 输出:π保留2位小数:3.14 // 2. 进制转换(十进制/八进制/十六进制) cout << "十进制:" << dec << num << endl; // 255 cout << "八进制:" << oct << num << endl; // 377 cout << "十六进制:" << hex << uppercase << num << endl; // FF(大写) // 3. 对齐与补零(宽度5,左对齐,补0) cout << "左对齐补0:" << setw(5) << left << setfill('0') << str << endl; // 输出:左对齐补0:abc00 // 4. 恢复默认格式 cout << resetiosflags(ios::fixed | ios::uppercase); return 0; }

核心格式化操纵符(来自<iomanip>

操纵符作用
setprecision(n)设置浮点数精度(n 为有效数字 / 小数位数,配合 fixed 为小数位数)
fixed浮点数固定小数格式
scientific浮点数科学计数法格式
setw(n)设置输出宽度(仅对下一个输出有效)
setfill(c)设置填充字符(默认空格)
left/right左对齐 / 右对齐(默认右对齐)
dec/oct/hex十进制 / 八进制 / 十六进制
uppercase十六进制 / 科学计数法大写输出
resetiosflags恢复格式化标志为默认

逐类详解:所有格式化场景

1. 浮点数格式化
操纵符组合效果底层逻辑
setprecision(n)有效数字 n 位(默认模式)流状态:ios::defaultfloat,截断 / 四舍五入到 n 位有效数字
fixed + setprecision(n)小数位 n 位(固定格式)流状态:ios::fixed,强制显示小数点,小数部分保留 n 位
scientific + setprecision(n)科学计数法,小数位 n 位流状态:ios::scientific,格式为尾数e±指数,尾数保留 n 位小数

完整示例

int main() { double pi = 3.141592653589793; double num = 1234.5678; // 场景1:默认模式(有效数字,默认6位) cout << "【默认模式】pi = " << pi << endl; // 输出:3.14159(6位有效数字) cout << "【默认模式】num = " << num << endl; // 输出:1234.57(6位有效数字,四舍五入) // 场景2:修改有效数字为4位(永久生效,直到重置) cout << "【有效数字4位】pi = " << setprecision(4) << pi << endl; // 3.142(4位有效数字) cout << "【有效数字4位】num = " << num << endl; // 1235(4位有效数字,四舍五入) // 场景3:固定小数位模式(保留2位小数) cout << "【固定2位小数】pi = " << fixed << setprecision(2) << pi << endl; // 3.14 cout << "【固定2位小数】num = " << num << endl; // 1234.57 // 场景4:科学计数法(保留3位小数) cout << "【科学计数法】pi = " << scientific << setprecision(3) << pi << endl; // 3.142e+00 cout << "【科学计数法】num = " << num << endl; // 1.235e+03 // 场景5:重置所有浮点数格式(恢复默认) cout << "【重置后】pi = " << resetiosflags(ios::fixed | ios::scientific) << setprecision(6) << pi << endl; // 3.14159(恢复默认6位有效数字) return 0; }
2. 整数格式化(对齐、补零、进制)
(1)进制转换(永久生效)
操纵符效果底层状态示例(num=255)
dec十进制ios::dec255
oct八进制ios::oct377
hex十六进制ios::hexff
uppercase十六进制 / 科学计数法大写ios::uppercaseFF
(2)对齐与补位(核心坑点:setw 仅对下一个输出有效)
操纵符组合效果注意事项
setw(n)设置输出宽度 n(仅下一个有效)不足宽度时补空格,超出宽度时按实际输出(不会截断)
setfill(c)设置填充字符(永久生效)必须配合setw使用,默认填充空格
left/right左 / 右对齐(永久生效)默认右对齐

完整示例(含坑点演示)

int main() { int id = 123; int num = 255; // 场景1:学号补零(宽度5,右对齐,补0) cout << "【学号补零】" << setw(5) << setfill('0') << id << endl; // 00123 // 坑点:setw仅对下一个有效,第二次输出id不会补零 cout << "【无setw】" << id << endl; // 123(无补零) // 场景2:宽度8,左对齐,补* cout << "【左对齐补*】" << setw(8) << left << setfill('*') << id << endl; // 123***** // 验证:left永久生效,下一个输出仍左对齐 cout << "【左对齐延续】" << setw(8) << num << endl; // 255***** // 场景3:十六进制大写输出 cout << "【十六进制大写】" << hex << uppercase << num << endl; // FF // 恢复十进制+默认对齐+填充空格 cout << resetiosflags(ios::uppercase) << dec << right << setfill(' '); // 场景4:宽度不足时不截断(重要) cout << "【宽度不足】" << setw(2) << 1234 << endl; // 1234(不会截断为34) return 0; }
3. 布尔值格式化
操纵符效果示例(flag=true)
boolalpha输出 true/falsetrue
noboolalpha恢复输出 1/0(默认)1

示例

int main() { bool flag = true; cout << "【默认布尔值】" << flag << endl; // 1 cout << "【boolalpha】" << boolalpha << flag << endl; // true cout << "【恢复默认】" << noboolalpha << flag << endl; // 1 return 0; }
4. 字符串格式化(对齐、补位)

字符串格式化逻辑与整数一致,重点注意setw仅对下一个字符串生效:

int main() { string name = "张三"; // 宽度10,右对齐,补- cout << "【右对齐补-】" << setw(10) << setfill('-') << name << endl; // --------张三 // 宽度10,左对齐,补- cout << "【左对齐补-】" << left << setw(10) << name << endl; // 张三-------- return 0; }

四、cout 性能优化

cout 默认会同步 C 语言的printf缓冲区,导致输出速度比printf慢很多,可通过以下两行代码优化(性能接近printf):

#include <iostream> using namespace std; int main() { // 核心优化:关闭C/C++流同步 + 解绑cin与cout ios::sync_with_stdio(false); cin.tie(nullptr); // 大量输出测试(优化后速度提升10倍+) for (int i = 0; i < 1000000; ++i) { cout << i << '\n'; // 用\n代替endl(避免频繁刷新缓冲区) } return 0; }

优化关键说明

  1. ios::sync_with_stdio(false):关闭cout与 C 标准输出的同步,减少开销;
  2. cin.tie(nullptr):解绑cincout(默认cin读取前会刷新cout缓冲区);
  3. \n代替endlendl会强制刷新缓冲区,大量输出时效率极低。

五、cout 扩展:自定义类型输出

通过重载operator<<运算符,可让cout直接输出自定义类 / 结构体:

#include <iostream> #include <string> using namespace std; // 自定义结构体:学生 struct Student { string name; int age; double score; }; // 重载<<运算符(核心步骤) ostream& operator<<(ostream& os, const Student& stu) { // 自定义输出格式 os << "姓名:" << stu.name << ",年龄:" << stu.age << ",成绩:" << stu.score; return os; // 返回os以支持链式输出 } int main() { Student s{"张三", 18, 95.5}; // 直接输出自定义类型 cout << s << endl; // 输出:姓名:张三,年龄:18,成绩:95.5 return 0; }

六、cout 常见问题与解决

  1. 问题 1:输出乱码(如中文乱码)
    • 解决:确保控制台编码与源码编码一致(如 UTF-8),Windows 可执行system("chcp 65001");
  2. 问题 2setw仅对下一个输出有效
    • 解决:如需多个输出保持宽度,需重复使用setw,或先格式化字符串再输出。
  3. 问题 3:浮点数输出精度不符合预期
    • 解决:配合fixed使用setprecision,明确指定小数位数。

总结

  1. cout 是 C++ 类型安全的标准输出工具,核心语法是cout << 内容endl换行并刷新缓冲区(推荐用\n提升性能);
  2. 格式化输出需包含<iomanip>,使用setprecision/setw等操纵符,语法比printf繁琐但类型安全;
  3. 性能优化关键:关闭ios::sync_with_stdio(false) + 解绑cin.tie(nullptr) + 用\n代替endl
  4. 支持自定义类型输出,只需重载operator<<运算符。

Read more

《算法闯关指南:动态规划算法--斐波拉契数列模型》--01.第N个泰波拉契数,02.三步问题

《算法闯关指南:动态规划算法--斐波拉契数列模型》--01.第N个泰波拉契数,02.三步问题

🔥草莓熊Lotso:个人主页 ❄️个人专栏: 《C++知识分享》《Linux 入门到实践:零基础也能懂》 ✨生活是默默的坚持,毅力是永久的享受! 🎬 博主简介: 文章目录 * 前言: * 01.第N个泰波拉契数 * 解法(动态规划): * 算法流程: * C++算法代码: * 算法总结&&笔记展示: * 02.三步问题 * 解法(动态规划): * 算法思路: * C++算法代码: * 算法总结&&笔记展示: * 结尾: 前言: 聚焦算法题实战,系统讲解三大核心板块:优选算法:剖析动态规划、二分法等高效策略,学会寻找“最优解”。 递归与回溯:掌握问题分解与状态回退,攻克组合、排列等难题。 贪心算法:理解“

By Ne0inhk

Flutter 三方库 rbush 的鸿蒙化适配指南 - 在鸿蒙系统上构建极致、极速、基于 R-Tree 算法的工业级 2D 空间索引与大规模点位碰撞检测引擎

欢迎加入开源鸿蒙跨平台社区:https://openharmonycrossplatform.ZEEKLOG.net Flutter 三方库 rbush 的鸿蒙化适配指南 - 在鸿蒙系统上构建极致、极速、基于 R-Tree 算法的工业级 2D 空间索引与大规模点位碰撞检测引擎 在鸿蒙(OpenHarmony)系统的地理信息系统(GIS)、高性能 2D 游戏引擎、或者是需要对成千上万个矩形/点位进行毫秒级“范围查询(Range Search)”的应用中,如何摆脱由于线性扫描(O(n))导致的由于由于性能瓶颈?rbush 为开发者提供了一套工业级的、基于高级 R-Tree 算法。管理过程。由于由空间索引方案。本文将深入实战其在鸿蒙业务逻辑层空间数据处理中的应用。 前言 什么是 RBush?它是一个旨在解决“如何快速在一个平面找到一群物体”的高性能数据结构。它采用了 R-Tree

By Ne0inhk
coding ability 展开第四幕(滑动指针——巩固篇)超详细!!!!

coding ability 展开第四幕(滑动指针——巩固篇)超详细!!!!

文章目录 * 前言 * 水果成篮 * 思路 * 找到字符串中所有字母异位词 * 思路 * 串联所有单词的子串 * 思路 * 最小覆盖子串 * 思路 * 总结 前言 本专栏上一篇博客,带着大家从认识滑动窗口到慢慢熟悉 相信大家对滑动窗口已经有了大概的认识 其实主要就是抓住——一段连续的区间 今天来学习一些滑动窗口进阶的题目 fellow me 水果成篮 思路 一开始看到这个题目,一段连续的区间,想到了滑动窗口 然后就想着怎么维护窗口,每次更新到新的水果种类就要,开始对left++,然后处理数据 其实是有点麻烦的,但是经过半个多小时的调试,最后还是ac了 思路:每次更新两个种类的水果,x,y,如果下一个水果的种类不相符合,就更新新的x,y 这个时候 right - 1 和 right 所对应的水果就是新的两种,然后就是处理从 left 到 right

By Ne0inhk