C++文件读写操作笔记

文件操作基础

C++通过<fstream>头文件提供文件操作支持,主要包含三个关键类:

  • ifstream:输入文件流(读取)
  • ofstream:输出文件流(写入)
  • fstream:双向文件流

文件打开模式:

|-------------|------------------|
| 模式标志 | 描述 |
| ios::in | 读取模式(默认ifstream) |
| ios::out | 写入模式(默认ofstream) |
| ios::app | 追加写入(不覆盖原有内容) |
| ios::binary | 二进制模式 |
| ios::trunc | 清空文件(默认out模式) |

文件读写

文本写入操作:

复制代码
void writeToFile(std::string fileName) {
    std::ofstream ofs(fileName, std::ios::app); // 追加写

    if (ofs) {
        for (int i = 0; i < 5; i++) {
            ofs << "content" << i << " ";
        }
    }
    else {
        std::cerr << "文件打开失败";
    }

    ofs.close();
}

文本逐行读取操作:

复制代码
void ReadLinetFile(std::string& fileName) {
	std::ifstream ifs(fileName);
	if (!ifs.is_open()) {
		std::cerr << "文件打开失败";
		return;
	}

	std::string line;
    
    // 以换行为结束标志
	while (std::getline(ifs, line)) {
		std::cout << line << std::endl;
	}
    ifs.clear();
    ifs.seekg(0, std::ios::beg);

    // 以空格或换行为结束标志
    std::string str;
    while (ifs >> str) {
        std::cout << str << std::endl;
    }

	ifs.close();
}

|-----------------------------|-----------------------------------------|
| 函数/操作 | 功能 |
| ifs.clear() | 清除failbit/eofbit 等错误标志(必须调用才能恢复正常的指针操作) |
| ifs.seekg(0, std::ios::beg) | 将输入位置指针定位到文件开头(字节偏移量为0) |
| ifs.tellg() | 获取当前读取位置(调试时可用来验证指针位置) |

二进制写操作:

复制代码
struct Record {
int id;
char name[20];
double value;
};
void writeToBinary(std::string fileName) {
    std::ofstream ofs(fileName, std::ios::binary);

    Record record = { 1, "hh", 2.32 };

    // reinterpret_cast<char*>(&record)
    if (ofs.write(reinterpret_cast<char*>(&record), sizeof(Record))) {
        std::cout << "success";
    }
    else {
        std::cerr << "error";
    }
}

reinterpret_cast<char*>(&record) 是C++中一种特定场景下的类型转换操作,用于直接将内存中的二进制数据视为字符流处理。

二进制读取操作:

复制代码
void readBinaryFile(const std::string& filename) {
    std::ifstream file(filename, std::ios::binary);

    if (!file) {
        std::cerr << "Error opening binary file" << std::endl;
        return;
    }

    Record record;
    while (file.read(reinterpret_cast<char*>(&record), sizeof(Record))) {
        std::cout << "ID: " << record.id
            << ", Name: " << record.name
            << ", Value: " << record.value << '\n';
    }
}

使用示例

从data.txt文件中读取前100个int整数:

复制代码
std::vector<int> read_first_100_text_ints(const std::string& filename) {
    std::ifstream file(filename);
    if (!file) {
        std::cerr << "Error opening file " << filename << std::endl;
        return {};
    }
    std::vector<int> result;
    result.reserve(100);
    int num;
    for (int i = 0; i < 100; ++i) {
        if (file >> num) {
            result.push_back(num);
            std::cout << num << " ";
        }
        else {
            if (file.eof()) {
                std::cout << "Reached end of file at number " << i << std::endl;
            }
            else {
                std::cerr << "Invalid data format at position " << i << std::endl;
                file.clear(); // 清除错误状态
                file.ignore(std::numeric_limits<std::streamsize>::max(), ' ');
            }
            break;
        }
    }
    return result;
}

file.ignore(std::numeric_limits<std::streamsize>::max(), ' ');

  • 定位错误 :通过i标识出错的读取操作序号(第i+1次尝试读取)
  • 流状态恢复file.clear()清除failbit等错误标志
  • 错误数据跳过ignore()跳过直到下一个空格或文件结尾

常见错误排查

|------------|-------------|-----------------|
| 错误现象 | 可能原因 | 解决方案 |
| 返回空数据 | 文件未成功打开 | 检查is_open()状态 |
| 二进制数据损坏 | 未指定binary模式 | 添加ios::binary标志 |
| 部分写入失败 | 磁盘空间不足 | 检查磁盘状态 |
| 读取和写入内容不匹配 | 平台换行符差异 | 使用'\n'代替endl |

相关推荐
阿斌_bingyu70921 分钟前
Arduino开发物联网ESP32快速入门指南(包含开发语言说明、学习路径和实战教程)
开发语言·物联网·学习
东方芷兰29 分钟前
JavaWeb 课堂笔记 —— 08 请求响应
xml·java·笔记·spring·tomcat·html·idea
Jwoka39 分钟前
正则表达式学习笔记
笔记·python·正则表达式·re
这个懒人1 小时前
linux下io操作详细解析
开发语言·c++·io
东方醴歌1 小时前
VMware安装飞牛私有云fnOS并挂载小雅Alist实现异地远程访问
开发语言·后端·golang
暗影~行星1 小时前
C语言,原码、补码、反码
c语言·开发语言
晨曦5432102 小时前
python——正则表达式
开发语言·正则表达式
NULL指向我2 小时前
TMS320F28P550SJ9学习笔记14:EPWM_死区dead_baund
笔记·学习
铃煦2 小时前
《算法笔记》3.3小节——入门模拟->图形输出
笔记·算法·图论
傍晚冰川2 小时前
【STM32】解读启动文件startup_stm32f10x_md.s
linux·笔记·stm32·单片机·学习