C++中二进制文件读写需使用std::ofstream和std::ifstream以std::ios::binary模式操作,通过write()和read()函数直接存取内存字节,避免字符转换,确保数据原样存储与读取。

在C++中将二进制数据写入文件,核心做法是利用
std::ofstream
std::ios::binary
write()
要实现C++中的二进制文件写入,我们通常会用到
<fstream>
std::ofstream
std::ios::binary
\n
\r\n
例如,如果你想写入一个整数或者一个自定义结构体,可以这样做:
#include <fstream>
#include <iostream>
#include <vector>
// 假设我们有一个这样的结构体
struct MyData {
int id;
double value;
char name[20];
};
int main() {
std::ofstream outFile("data.bin", std::ios::out | std::ios::binary);
if (!outFile.is_open()) {
std::cerr << "错误:无法打开文件进行写入!" << std::endl;
return 1;
}
// 写入一个整数
int anInteger = 12345;
// write方法的第一个参数是char*类型,所以需要reinterpret_cast
outFile.write(reinterpret_cast<const char*>(&anInteger), sizeof(anInteger));
// 写入一个浮点数
float aFloat = 3.14159f;
outFile.write(reinterpret_cast<const char*>(&aFloat), sizeof(aFloat));
// 写入自定义结构体
MyData myRecord = {1, 99.9, "TestRecord"};
outFile.write(reinterpret_cast<const char*>(&myRecord), sizeof(myRecord));
// 写入一个字节数组(或std::vector<char>)
std::vector<char> byteBuffer = {'A', 'B', 'C', 0x01, 0x02, 0x03};
outFile.write(byteBuffer.data(), byteBuffer.size());
outFile.close(); // 养成好习惯,手动关闭文件
std::cout << "二进制数据已成功写入 data.bin" << std::endl;
return 0;
}这里需要特别注意
reinterpret_cast<const char*>(&data)
write()
const char*
const char*
sizeof()
sizeof(data)
立即学习“C++免费学习笔记(深入)”;
既然能写,那自然也要能读。在C++中高效地读取二进制文件内容,我们主要依赖
std::ifstream
read()
#include <fstream>
#include <iostream>
#include <vector>
// 沿用之前的结构体定义
struct MyData {
int id;
double value;
char name[20];
};
int main() {
std::ifstream inFile("data.bin", std::ios::in | std::ios::binary);
if (!inFile.is_open()) {
std::cerr << "错误:无法打开文件进行读取!" << std::endl;
return 1;
}
// 读取之前写入的整数
int readInteger;
inFile.read(reinterpret_cast<char*>(&readInteger), sizeof(readInteger));
if (inFile.gcount() == sizeof(readInteger)) { // 检查是否读取了预期数量的字节
std::cout << "读取的整数: " << readInteger << std::endl;
} else {
std::cerr << "读取整数失败或不完整。" << std::endl;
return 1;
}
// 读取浮点数
float readFloat;
inFile.read(reinterpret_cast<char*>(&readFloat), sizeof(readFloat));
if (inFile.gcount() == sizeof(readFloat)) {
std::cout << "读取的浮点数: " << readFloat << std::endl;
} else {
std::cerr << "读取浮点数失败或不完整。" << std::endl;
return 1;
}
// 读取自定义结构体
MyData readRecord;
inFile.read(reinterpret_cast<char*>(&readRecord), sizeof(readRecord));
if (inFile.gcount() == sizeof(readRecord)) {
std::cout << "读取的结构体ID: " << readRecord.id
<< ", Value: " << readRecord.value
<< ", Name: " << readRecord.name << std::endl;
} else {
std::cerr << "读取结构体失败或不完整。" << std::endl;
return 1;
}
// 读取剩余的字节数组
// 假设我们知道文件剩余大小,或者循环读取直到文件结束
// 这里我们简单读取固定大小的字节
std::vector<char> readBuffer(6); // 之前写入了6个字节
inFile.read(readBuffer.data(), readBuffer.size());
if (inFile.gcount() == readBuffer.size()) {
std::cout << "读取的字节数组: ";
for (char c : readBuffer) {
// 注意:直接输出char可能会显示为字符或乱码,这里转换为int看其数值
std::cout << static_cast<int>(c) << " ";
}
std::cout << std::endl;
} else {
std::cerr << "读取字节数组失败或不完整。" << std::endl;
return 1;
}
inFile.close();
return 0;
}在实际应用中,尤其是在处理大型文件时,你可能会一次性读取一个较大的缓冲区,而不是一个一个地读取小数据块。例如,可以先获取文件大小,然后分配一个足够大的
std::vector<char>
read()
gcount()
read()
二进制文件操作虽然直接,但“直接”也意味着你得自己处理更多底层细节。我个人在处理这类问题时,遇到过不少坑,其中有些确实让人头疼。
首先,字节序(Endianness) 是个大问题。你在一台小端序(Little-endian)机器上写入的
int
0x12345678
78 56 34 12
78 56 34 12
0x78563412
其次,结构体内存对齐和填充(Padding) 也常常被忽视。C++编译器为了提高访问效率,可能会在结构体成员之间插入一些空白字节(padding)。这意味着
sizeof(MyData)
sizeof()
sizeof(MyData)
#pragma pack(1)
__attribute__((packed))
再来就是错误处理不充分。很多新手会忘记检查
is_open()
fail()
bad()
eof()
reinterpret_cast
最后,文件关闭。虽然
std::ofstream
std::ifstream
close()
在我看来,C++二进制文件操作与文本文件操作的核心区别,就像是机器语言和自然语言的区别:一个追求效率和精确的底层表达,另一个则更注重人类的阅读和理解。
最直观的区别在于数据的解释方式。
123
'1'
'2'
'3'
\n
\r\n
123
123
int
0x00 0x00 0x00 0x7B
std::ios::binary
这种解释方式的不同,直接导致了几个关键差异:
总的来说,如果你需要存储大量、复杂、结构化的数据,并且对性能和存储空间有要求,那么二进制文件是更好的选择。而如果数据主要是给人阅读,或者数据量不大,且需要方便地用文本工具处理,那么文本文件就更合适。这两种方式各有千秋,选择哪种取决于你的具体需求和应用场景。
以上就是如何在C++中将二进制数据写入文件_C++文件二进制读写操作的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号