17370845950

c++中如何逐行读取文件_c++ getline读取文件每一行【实例】
std::getline是逐行读取文件最稳妥的方式,能完整保留每行原始内容(不含换行符),推荐循环条件为while (getline(file, line)),需检查文件是否成功打开。

std::getline 逐行读取文件是最稳妥的方式

直接用 operator>> 读字符串会跳过空白、无法处理含空格的整行,而 std::getline 按换行符切分,能完整保留每行原始内容(不含换行符本身)。前提是文件以文本模式打开,且流状态正常。

  • std::ifstream 必须检查是否成功打开:if (!file.is_open())if (!file)
  • 不要在 while (getline(...)) 外提前调用 file.good() 判断——它可能为 true,但下一次 getline 仍失败
  • Windows 换行符 \r\n 会被 std::getline 自动识别并截断 \n,无需额外处理

标准写法:while (getline(file, line))

是唯一推荐循环条件

这个条件既检查读取是否成功,又自然处理 EOF 和读取失败(如磁盘错误)。用 !file.eof()file.good() 做循环判断会导致最后一行重复或漏读。

std::ifstream file("data.txt");
std::string line;
while (std::getline(file, line)) {
    std::cout << "Line: [" << line << "]\n";
}
if (file.bad()) {
    std::cerr << "I/O error occurred.\n";
}

读取后需要解析数字?小心 std::stoi 抛异常

如果某行是纯数字(如 "42"),可直接用 std::stoi;但若含前导空格或非数字字符(如 " 42abc"),std::stoi 会截断解析并静默忽略尾部,而 "abc" 会抛 std::invalid_argument。更安全的做法是先 trim 再校验:

  • line.find_first_not_of(" \t")line.find_last_not_of(" \t") 手动去首尾空白
  • 解析前用 line.empty() || line.find_first_of("0123456789-+") == std::string::npos 粗筛(不完美但快)
  • 真正健壮的数字解析建议用 std::from_chars(C++17 起),它不抛异常、返回解析长度

大文件性能注意:std::string 复用比反复构造更快

每次循环中声明 std::string line; 看似简洁,但频繁分配/释放内存会影响吞吐。对 GB 级日志类文件,应复用同一 std::string 对象,并在循环开始前调用 line.clear() 或直接依赖 getline 的覆盖行为(它会重置长度并按需扩容)。

真正卡顿往往不在读取本身,而在后续处理——比如每行都做正则匹配或网络请求。把 I/O 和计算解耦(例如先批量读入 vector,再另起线程处理)才是关键优化点。