避免c++缓冲输出中的内存占用

Avoiding memcpy in C++ buffered output

本文关键字:内存 c++ 缓冲 输出 避免      更新时间:2023-10-16

我正试图直接写入fstream输出缓冲区以避免memcpy

为什么下面的代码不能工作?

在Linux上编译、运行并产生正确长度的输出文件。但是输出文件不包含正确的文本。还请注意,由于某种原因,当我注释掉涉及str2的两行时,生成了一个长度为零的输出文件。

注意:这个例子没有避免memcpy,但如果它工作,它将帮助我在我的应用程序中避免memcpy

#include <fstream>
int main(int argc, char *argv[]) {
  std::fstream out;
  char buffer[512];
  out.rdbuf()->pubsetbuf(buffer, 512);
  out.open("file.txt", std::fstream::out);
  char *str1 = "test text.";
  strcpy(buffer, str1);
  out.rdbuf()->pubseekpos(strlen(str1), std::ios_base::out);
  char *str2 = "why?";
  out << str2;
  out.flush();
  out.close();
}

给流一个缓冲区供内部使用。那你就别给它写东西了

在不告诉流的情况下将某些内容复制到缓冲区,这不会在文件中给你任何东西。

您可能已经注意到,seekpos用于在文件中定位,而不是在缓冲区中移动。缓冲区仅供流内部使用!

当您执行out.rdbuf()->pubseekpos(strlen(str1), std::ios_base::out);时,这将要求流向前跳过新创建的文件。你能指望它从你指定的缓冲区中获取东西吗?提示:您是否见过任何涉及指定缓冲区的示例,说明您需要以某种方式初始化它或指定它包含的非垃圾字符的初始数量?不…因为流本身跟踪缓冲区的哪些部分正在使用。因此,当您使用假定的空缓冲区并向前跳过时,它会生成中间的null。将它们设置在缓冲区中是没有意义的(所以简单地在pubseekpos之后执行memcpy也行不通)-如果您向前跳了超过缓冲区大小的值怎么办?

希望这至少能说明问题,尽管我在这个阶段没有给出任何想法,你可能会强迫流改变它的"跟踪"有意义的缓冲区内容....

现在还早,但你不是在做

#include <fstream>
int main(int argc, char *argv[]) 
{
  std::fstream out;
  out.open("file.txt", std::fstream::out);
  out << "test text";
  out << "why?";
  out.flush();
  out.close();
}

?

我对c++ iostream库不太熟悉,无法说出你的程序出了什么问题,我只想说,如果你想自己做缓冲,直接使用read()和write()接口可能会更直接(而不是write)。如果您自己做缓冲,iostream库可能不会给您带来太多好处,而且只会模糊实际发生的事情。

我认为有三种可能性

  1. 实现你自己的流缓冲类
  2. 使用本地文件API
  3. 禁用pubsetbuf(0, 0)缓冲

实现一个流缓冲区并没有那么困难,它可以让你有机会将c++风格的流插入和直接内存访问混合在一起,而不需要牺牲性能。