如何将文件流作为类成员

How to have a file stream as a class member

本文关键字:成员 文件      更新时间:2023-10-16

我有以下在Visual c++中工作的解析器类

class Parser
{
   private:
   const char* filename;
   std::ifstream filestream;
   std::vector<std::string> tokens;
   unsigned int linect;
   public:
   Parser(const char* filename);
   bool readline();
   std::string getstrtoken(unsigned int i) const { return tokens[i]; }
   int getinttoken(unsigned int i) const { return atoi(tokens[i].c_str()); }
};
Parser::Parser(const char* filename) :
   filename(filename),
   linect(0)
{
   filestream = ifstream(filename); // OK in VC++, not with GCC?
}
bool Parser::readline()
{
   std::string line;
   getline(filestream, line);
   std::stringstream ss(line);
   std::string token;
   tokens.clear();
   while(getline(ss, token, ' ')){ if(token != "") tokens.push_back(token); }
   linect++;
   return (filestream != NULL);
}

但是当我尝试用GCC 4.8.2编译它时,我得到错误说我不能分配给filestream。从我在这个网站的其他地方看到的,你可以做

std::ifstream filestream(filename);

但是你不能做

std::ifstream filestream;
filestream = ifstream(filename);

如果我想将filestream声明为Parser类的成员并在构造函数中初始化它,这实际上是我需要做的。

我想把文件流保存在Parser类中,以便那些使用解析器的人不需要声明和跟踪它。在我看来,这应该是自包含在Parser类,因为它的内部方法(例如readline())是唯一使用它的。

是否有一种方法可以在两个平台上实现这一点?

谢谢。

edit:我的修复方法是显式调用ifstreamopen()方法。我的解析器类构造函数现在看起来像:

Parser::Parser(const char* filename) :
   filename(filename),
   linect(0)
{
   filestream.open(filename);
   // Do some checking to make sure the file exists, etc.
}

不能,因为std::ifstream已经删除了复制构造函数和复制赋值。你可以通过做

filestream.swap(ifstream(filename)).

它在visual studio上编译的事实主要是因为它被内联到move赋值或move构造函数(我不太好告诉你到底是哪一个)。如果你尝试

std::ifstream myF;
filestream = myF;

无法编译

然而,你可以尝试做我写的移动,或者你可以调用.open (http://en.cppreference.com/w/cpp/io/basic_ifstream/open)

我认为更好的解决办法是:

  1. 先构建ifstream
  2. 使用ifstream对象构建Parser
  3. 更改Parser以存储对istream对象的引用。这使您能够灵活地解析文件、标准输入和字符串的内容。

class Parser
{
   private:
      std::istream& str;
      std::vector<std::string> tokens;
      unsigned int linect;
   public:
      Parser(std::istream& s) : str(s) ... {}
      ...
};

std::ifstream没有复制构造函数,可能是vc++的众多扩展之一。正确的代码是:

Parser::Parser(const char* filename) :
   filename(filename),
   linect(0),
   filestream(filename)
{
}

请注意成员变量和参数filename使用this->或更改名称(建议,通常前缀用于成员变量_m_)