c++ - rdbuf() 奇怪的行为

标签 c++ string file buffer

我试图通过简单地使用 rdbuf() 将两个文件合并为一个文件,但是在合并时,文件的第一行丢失了。代码如下:

void mergeFiles(ifstream &file1, ifstream &file2, ofstream &file3) {

    int num1;
    int num2;

    string firstLine_1;
    string firstLine_2;

    getline(file1, firstLine_1);
    num1 = atoi(firstLine_1.c_str());

    getline(file2, firstLine_2);
    num2 = atoi(firstLine_2.c_str());

    if (num1<num2) {
        file3 << file1.rdbuf() << "\n" << file2.rdbuf();
    } else {
        file3 << file2.rdbuf() << "\n" << file1.rdbuf();
    }
}

最佳答案

好吧,您在连接其余文件之前先取出每个文件的第一行,这是预期的。不过,你仍然有这些行,所以只需在它们之间打印它们:

if (num1<num2) {
    file3 << firstLine_1   << "\n"
          << file1.rdbuf() << "\n"
          << firstLine_2   << "\n"
          << file2.rdbuf();
} else {
    file3 << firstLine_2   << "\n"
          << file2.rdbuf() << "\n"
          << firstLine_1   << "\n"
          << file1.rdbuf();
}

附录: @BenVoigt 正确地指出,如果其中一个文件包含不以换行符结尾的单行,则其输出将不同于打印由单个换行符(中间或末尾会有一个额外的换行符)。

基于这种情况是可识别的,因为第一个 getline 调用将设置文件结束标志,这可以通过以下方式避免:

if (num1<num2) {
    file3 << firstLine_1;
    if(file1) { file3 << "\n" << file1.rdbuf(); }
    file3 << "\n";
    file3 << firstLine_2;
    if(file2) { file3 << "\n" << file2.rdbuf(); }
} else {
    file3 << firstLine_2;
    if(file2) { file3 << "\n" << file2.rdbuf(); }
    file3 << "\n";
    file3 << firstLine_1;
    if(file1) { file3 << "\n" << file1.rdbuf(); }
}

由于这是冗长且重复的内容,我会考虑使用 lambda 使其更简洁:

auto printfile = [&file3](std::string const &firstline, std::istream &in) {
  file3 << firstline;
  if(in) { file3 << "\n" << in.rdbuf(); }
};

if(num1 < num2) {
  printfile(firstLine_1, file1);
  file3 << "\n";
  printfile(firstLine_2, file2);
} else {
  printfile(firstLine_2, file2);
  file3 << "\n";
  printfile(firstLine_1, file1);
}

对于像这样的常规文件,回溯也是一种可能,但我觉得最好避免在没有令人信服的理由的情况下放弃在管道、套接字和其他不可寻找的流上使用代码的能力,即使看起来就像一个不太可能的用例。

关于c++ - rdbuf() 奇怪的行为,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28730507/

相关文章:

c++ - 在 SDL2 和 C++ 中编码图像和改变颜色

c++ - 具有源子目录的单个顶级 makefile

c++ - 如何比较两个包含十进制值的字符串?

c# - 如何在txt的新行中添加每个字典的值?

c++ - 错误 undefined reference `BP::Device::Create(std::string const&)'

C++ - 使控制台全屏显示?

Java ArrayList 项目计数器 ie "Apple, 3×Banana, 2×Orange"

java - 从文件读取时的字符串格式

excel - VBA 集合检索数据但不显示值

c++ - 从 C++ 控制台应用程序上传 txt 文件