简单搞懂 C++ 文件读写:`std::ofstream` 和 `std::ifstream`

文件读写是编程的必备技能。在 C++ 中,用 std::ofstream 写文件,std::ifstream 读文件。

写入文件(std::ofstream

1)基础写入

#include <fstream>

// 1. 创建并打开文件(默认清空原内容)
std::ofstream fout("data.txt"); // 默认模式:覆盖写入(若文件存在则清空,否则创建)

// 2. 检查是否成功打开
if (!fout.is_open()) { 
    std::cerr << "文件打开失败!";
    return;
}

// 3. 写入内容(文本模式自动处理换行符)
fout << "Hello, 你好!\n";
fout << 42 << " " << 3.14; // 写入数字

// 4. 关闭文件(不写也会自动关闭)
fout.close(); 

2)二进制模式写入

// 写入二进制数据(如结构体)
struct Point { int x, y; };
Point p{10, 20};

std::ofstream binFile("point.bin", std::ios::binary); // 二进制方式
binFile.write(reinterpret_cast<char*>(&p), sizeof(Point));

3)预分配文件空间

// 目标:创建一个 1024 字节的空文件
std::ofstream file("prealloc.bin", std::ios::binary);
file.seekp(1023);    // 移动到第 1023 字节位置(从0开始)
file.put('\0');      // 写入1字节,总大小变为 1024

seekp(N-1) + put('\0') 才能生成准确大小为 N 的文件。

读取文件(std::ifstream

1)基础读取

#include <fstream>
#include <string>

std::ifstream fin("data.txt"); // 默认只读文本模式
if (!fin) { // 检查是否打开成功
    std::cerr << "文件打开失败!";
    return;
}

// 逐行读取(推荐!)
// 使用 std::string
std::string line;
while (std::getline(fin, line)) {
    std::cout << line << std::endl;
}

// 使用 char[]
char buffer[256];
while (fin.getline(buffer, sizeof(buffer))) {
    std::cout << buffer << std::endl;
}

// 按词读取
std::string word;
while (fin >> word) {  // 以空格为分隔符读取单词
    std::cout << word << " ";
}

// 一次性读取全文
fin.seekg(0, std::ios::end);  // 移动指针到末尾
size_t size = fin.tellg();    // 获取文件大小
fin.seekg(0);                 // 回到文件头
std::string content(size, ' '); //创建与文件大小相同的string对象content,并用空格初始化
fin.read(&content[0], size);  // 读取全部内容到content中

2)二进制模式读取

// 读取结构体
Point p;
std::ifstream binFile("point.bin", std::ios::binary);
binFile.read(reinterpret_cast<char*>(&p), sizeof(Point));

重点注意

1)不要重复打开文件:重复打开文件会清空内容。

2)文本模式 vs 二进制模式

  • 文本模式(默认):自动转换换行符(如 Windows 的 \n\r\n\r\n\n)。
  • 二进制模式:直接读写原始字节,不转换。
  • 适用场景
    • 文本文件用默认模式。
    • 图片、结构体数据用二进制模式。

3)不要用 eof() 做循环条件

  • eof()(End-Of-File)仅在读取操作尝试越过文件末尾时才会被设置为真。可能导致读取两次末尾数据,第二次读取末尾数据时读取失败,eof() 变为 true,存在重复处理末尾数据的可能。
  • getline 的机制:如果读取失败(如到达文件末尾),buffer 不会被修改。

总结

  • 写文件:<<write()
  • 读文件:>>getline()read()

常用模式总结表

模式名称模式标志行为描述
覆盖std::ios::out默认模式:文件存在则清空内容,不存在则创建;写入新内容。
追加std::ios::out std::ios::app写入新内容到文件末尾。
二进制std::ios::out std::ios::binary以二进制格式写入数据。
文本std::ios::in仅读取文本文件内容;文件必须存在,否则打开失败。
二进制std::ios::in std::ios::binary以二进制格式读取数据。

文件指针定位方式总结表

方法签名基准位置行为描述
seekg(offset, std::ios::beg)文件开头从文件开头绝对定位到 offset 字节处(负数无效)
seekg(offset, std::ios::cur)当前位置从当前位置相对移动 offset 字节(正数向前,负数向后)
seekg(offset, std::ios::end)文件末尾从文件末尾绝对定位到 offset 字节处(正数无效,一般用负数)
seekg(offset)(单参数形式)当前位置等价于 seekg(offset, std::ios::cur),但只能向前移动
seekp(offset, origin)seekg写指针定位,行为与 seekg 一致(仅用于 ofstreamfstream 的写操作)
<think>好的,我现在需要帮助用户了解如何在C++中使用文件流类,包括std::fstream、std::ifstreamstd::ofstream。首先,我得回忆一下这些类的基本用途区别。 用户提到了参考的引用内容,里面提到std::ofstream用于写文件std::ifstream用于读文件,而fstream是它们的基类,控制文件读写操作。这可能意味着用户需要具体的代码示例步骤说明。 首先,我应该分点讲解每个类的用法,包括头文件、构造函数、打开文件的方式,以及读写操作的具体方法。比如,使用std::ofstream写入文件时,可能需要说明如何打开文件、检查是否成功打开、写入数据,然后关闭文件。同样,对于std::ifstream,需要读取文件的步骤,比如打开文件、读取内容的方式(逐行读取或全部读取),以及错误处理。 接下来,可能需要提到fstream的读写模式,比如如何同时进行读写操作,这时候需要指定正确的打开模式,如std::ios::in | std::ios::out。此外,文件打开模式的其他选项,如app、ate、trunc等也需要解释清楚,避免用户混淆。 用户可能还需要了解如何检查文件是否成功打开,这在文件操作中非常重要,可以避免后续操作出错。例如,使用is_open()方法或者直接判断流的状态。 然后,考虑到用户请求的是教程或文档,我需要提供具体的代码示例,这样用户可以直接参考并实践。例如,写文件的代码示例,读文件的示例,以及使用fstream进行读写的示例。这些示例应该简洁明了,包含必要的注释,说明每一步的作用。 另外,需要注意常见的错误注意事项,比如文件路径的正确性,权限问题,以及打开模式的选择。例如,当用ofstream写入文件时,默认情况下会覆盖原有内容,如果用户需要追加,应该使用std::ios::app模式。 最后,生成相关问题,帮助用户进一步深入理解。这些问题可能涉及性能优化、异常处理、二进制文件操作,以及与其他数据格式的结合使用,比如JSON或XML。 现在需要确保回答结构清晰,按照读、写、读写的顺序组织内容,每个部分都有代码示例说明。同时,正确使用中文,避免技术术语错误,并引用提供的参考内容。</think>### C++文件流类使用指南 C++标准库通过`<fstream>`提供文件操作类:`std::ifstream`(读文件)、`std::ofstream`(写文件)、`std::fstream`(读写混合)。以下是具体使用方法: --- #### **1. 写入文件std::ofstream)** **步骤**: 1. 包含头文件:`#include <fstream>` 2. 创建对象并打开文件: ```cpp std::ofstream outFile("data.txt"); // 默认覆盖模式 // 追加模式:std::ofstream outFile("data.txt", std::ios::app); ``` 3. 检查是否成功打开: ```cpp if (!outFile.is_open()) { std::cerr << "文件打开失败"; return; } ``` 4. 写入数据: ```cpp outFile << "Hello, World!" << std::endl; outFile << 123 << " " << 45.6; ``` 5. 关闭文件:`outFile.close();` **示例**: ```cpp #include <fstream> int main() { std::ofstream file("output.txt"); if (file) { file << "写入内容"; file.close(); } return 0; } ``` --- #### **2. 读取文件std::ifstream)** **步骤**: 1. 创建对象并打开文件: ```cpp std::ifstream inFile("data.txt"); ``` 2. 逐行读取内容: ```cpp std::string line; while (std::getline(inFile, line)) { std::cout << line << std::endl; } ``` 3. 按数据类型读取: ```cpp int a; double b; inFile >> a >> b; ``` **示例**: ```cpp #include <fstream> #include <iostream> int main() { std::ifstream file("input.txt"); if (file) { std::string content; while (getline(file, content)) { std::cout << content << std::endl; } file.close(); } return 0; } ``` --- #### **3. 读写混合(std::fstream)** **用法**: ```cpp std::fstream ioFile("data.txt", std::ios::in | std::ios::out); // 需同时指定读写模式 if (ioFile) { ioFile << "新增内容"; ioFile.seekg(0); // 将读取指针移到文件开头 std::string line; getline(ioFile, line); ioFile.close(); } ``` --- #### **4. 文件打开模式** | 模式 | 说明 | |----------------|----------------------------------| | `std::ios::in` | 读模式(默认ifstream) | | `std::ios::out`| 写模式(默认ofstream,覆盖内容) | | `std::ios::app`| 追加写入 | | `std::ios::ate`| 打开后定位到文件末尾 | | `std::ios::binary` | 二进制模式 | --- #### **注意事项** - 文件路径需正确(绝对路径或相对路径)[^1]。 - 操作后及时关闭文件,避免资源泄露。 - 二进制文件需使用`read()`/`write()`方法。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值