C++中IO类(iostream、fstream和sstream)知识详解和应用

一、C++ I/O 类体系概览

C++ 的 I/O 功能由一组 流(stream) 类封装,位于头文件 <iostream><fstream><sstream> 等。核心类别及其继承关系简图如下:

复制代码
          ios_base
              ↑
       basic_ios<CharT,Traits>
              ↑
    ┌─────────┴─────────┐
    │                   │
basic_istream      basic_ostream
    ↑                   ↑
  istream             ostream
    ↑                   ↑
 ifstream, istringstream  ofstream, ostringstream
    ↑                   ↑
iostream               
  • istream:输入流;ostream:输出流
  • iostream:既能输入也能输出(如 std::cin/std::cout
  • ifstream/ofstream/fstream:文件流;
  • istringstream/ostringstream/stringstream:基于字符串缓冲的内存流。

二、常用流对象

对象 头文件 用途
std::cin <iostream> 从标准输入读取
std::cout <iostream> 向标准输出写入
std::cerr <iostream> 向标准错误写入(不带缓冲)
std::clog <iostream> 向标准错误写入(带缓冲)
std::ifstream <fstream> 从文件读取
std::ofstream <fstream> 向文件写入
std::fstream <fstream> 同时读写文件
std::istringstream <sstream> 从内存字符串读取
std::ostringstream <sstream> 向内存字符串写入
std::stringstream <sstream> 内存中读写混合

三、打开模式与文件流

cpp 复制代码
#include <fstream>

// 打开文件写入(若不存在则创建,若存在则截断)
std::ofstream out("data.txt", std::ios::out | std::ios::trunc);

// 追加写入
std::ofstream app("data.txt", std::ios::out | std::ios::app);

// 读取二进制
std::ifstream in("data.bin", std::ios::in | std::ios::binary);

// 读写
std::fstream fs("db.bin", std::ios::in|std::ios::out|std::ios::binary);
  • 常用模式

    • ios::inios::outios::app(尾部追加)、ios::trunc(截断)、ios::binary(二进制)
  • 检查打开状态

    cpp 复制代码
    if (!out.is_open()) {
        std::cerr << "无法打开文件\n";
    }

四、格式化与操纵器(Manipulator)

cpp 复制代码
#include <iostream>
#include <iomanip>

double x = 123.456789;

// 控制浮点精度
std::cout << std::fixed << std::setprecision(2) << x << "\n";  // 123.46

// 控制宽度与对齐
std::cout << std::setw(10) << std::left << "Hello" << "|\n";

// 控制进制
int n = 255;
std::cout << std::hex << n << " " << std::dec << n << "\n";    // ff 255

// 重置格式
std::cout << std::defaultfloat << std::right;
  • <iomanip> 中常用:setwsetfillsetprecisionfixedscientifichexdecboolalpha 等。

五、同步与性能注意事项

  1. 关闭同步

    cpp 复制代码
    std::ios::sync_with_stdio(false);
    std::cin.tie(nullptr);
    • 关闭与 C 标准库(stdio)的同步,可大幅提升 cin/cout 性能。
    • cin.tie(nullptr) 解除 cincout 的自动刷新绑定。
  2. 避免频繁刷新

    • std::endl 会刷新缓冲区,使用 '\n' 代替可减少开销。
  3. 缓冲区大小

    • 可通过自定义缓冲区(继承自 std::streambuf)或调用 rdbuf()->pubsetbuf() 调整。

六、错误处理与例外

  • 状态位rdstate() / good() / eof() / fail() / bad()

    cpp 复制代码
    if (in.fail()) { /* 读取失败 */ }
    if (in.eof())  { /* 到达文件末尾 */ }
  • 抛出例外

    cpp 复制代码
    in.exceptions(std::ios::failbit | std::ios::badbit);
    try {
        int v;
        in >> v;    // 失败时抛 ios_base::failure
    } catch (const std::ios_base::failure& e) {
        std::cerr << "I/O 错误: " << e.what() << "\n";
    }

七、stringstream 与文本解析

cpp 复制代码
#include <sstream>
#include <string>

std::string line = "100,3.14,hello";
std::istringstream ss(line);

int    a;
double b;
std::string s;

char comma;
ss >> a >> comma >> b >> comma >> s;
// a=100, b=3.14, s="hello"
  • 用于将字符串分割解析为各种类型,避免手写字符串处理逻辑。

八、自定义流缓冲区

  • 继承 std::streambuf 并重载底层读写函数,可实现内存、网络、压缩等自定义 I/O。
  • 示例略:需重载 underflow()(输入)、overflow()(输出)等。

九、示例:日志类

cpp 复制代码
#include <iostream>
#include <fstream>
#include <mutex>
#include <string>

class Logger {
    std::ofstream ofs;
    std::mutex    mtx;
public:
    Logger(const std::string& filename)
      : ofs(filename, std::ios::app)
    {
        if (!ofs) throw std::runtime_error("无法打开日志文件");
    }

    void log(const std::string& msg) {
        std::lock_guard<std::mutex> lk(mtx);
        ofs << msg << '\n';
        // 不使用 endl 以避免每次都 flush
    }
};

int main() {
    Logger log("app.log");
    log.log("程序启动");
    log.log("处理完成");
    return 0;
}

十、注意事项汇总

  1. 流关闭 :文件流在析构时自动关闭,也可显式调用 close()
  2. 资源泄露:异常安全时确保流对象析构,以正确释放文件句柄。
  3. 同步问题:多线程写同一流需加锁或使用线程安全的自定义缓冲。
  4. 格式状态:流对象的格式状态会保留,若在一处修改,需注意在别处可能受影响。
  5. Locale 设置 :对数字、小数点、宽字符支持等有要求时,可使用 std::locale 或调用 imbue()
  6. 二进制 I/O :读写 POD 类型时要注意对齐与字节序问题,推荐使用 read()/write()
相关推荐
王老师青少年编程5 小时前
csp信奥赛C++高频考点专项训练之贪心算法 --【哈夫曼贪心】:合并果子
c++·算法·贪心·csp·信奥赛·哈夫曼贪心·合并果子
叼烟扛炮6 小时前
C++第二讲:类和对象(上)
数据结构·c++·算法·类和对象·struct·实例化
样例过了就是过了7 小时前
LeetCode热题100 最长公共子序列
c++·算法·leetcode·动态规划
谭欣辰7 小时前
C++ 排列组合完整指南
开发语言·c++·算法
橙子也要努力变强8 小时前
信号捕捉底层机制-机理篇2
linux·服务器·c++
盐焗鹌鹑蛋8 小时前
【C++】stack和queue类
c++
郝学胜-神的一滴9 小时前
罗德里格斯旋转公式(Rodrigues‘ Rotation Formula)完整推导
c++·unity·godot·图形渲染·three.js·unreal
lzh200409199 小时前
深入理解进程:从PCB内核结构到写时拷贝的底层实战
linux·c++
aseity10 小时前
跨平台项目中QString 与 非Qt 跨平台动态库在字符集上的一个实用的互操作约定.
c++·经验分享
CN-Dust10 小时前
【C++】while语句例题专题
数据结构·c++·算法