报告/记录器类的流的 C++ 重载运算符<<

c++ overloading operator << of ofstream for Report/Logger class

本文关键字:lt 运算符 重载 记录器 报告 C++      更新时间:2023-10-16

我正在尝试用c ++创建一个报告/记录器类。

我想在主要有一个名为 Report 的对象,并将运算符<<应用于此类,就像 ofstream 一样写入文件多个字符串。

因此,而不是使用以下代码:

  ofstream myfile ("d://example.txt");
  if (myfile.is_open())
  {
    myfile << "This is a line.n" << "Heya!!!" << endl;
    myfile.close();
  }

我想要以下内容:

  Report rLogger ("d://example.txt"); // Report C'tor (const string& s)
  logger << "This is a line.n" << "Heya!!!" << endl;
// the destructor of rLogger will close the file when it dies...

我只是找不到一种方法将运算符<<编写为成员函数,这似乎正是我在这里所需要的。我可以使用运算符<<的唯一方法是作为朋友,这在这种情况下没有帮助。有人可以告诉我如何实现上述方法吗?

提前感谢,家伙

尝试以下操作:

class Report {
    public:
        Report() {}
        Report(std::string _str) : stream(_str) {}
        Report& operator <<(std::string str) {
            if (stream.is_open())
                stream << str;
            return *this;
        }
        ~Report() { stream.close(); }
        std::ofstream stream;
};

对于上述内容,您还必须包含<string>标头。

似乎您想创建一个类似流的类型。不要通过重载输出运算符来创建类似流的类型<<!创建新的类似流类型的方法是创建一个流缓冲区(即从std::streambuf派生的类型),并使用它来初始化std::ostream对象,可能通过使用std::ostream作为基类。

重载operator<<()的其他方法不适用于像std::endl这样的操纵器:操纵器std::endl是一个函数模板,为了在使用时推导它模板参数,必须有一个重载,可以用来推断类型正确。std::ostream类通过提供合适的特殊重载来实现这一点。

我意识到这并不能回答您关于如何超载operator<<()作为成员的问题,但我认为这是错误的方法,它可能会产生更好的结果,改变解决问题的方式。我会提供有关如何创建相应类的更多详细信息,但没有给出足够的上下文。

您的问题似乎源于需要在报告中插入不同类型的内容,例如 "Heya!!!"std::endl

一个灵活的解决方案是使用模板执行此操作。

您可以将其实现为成员函数,其中Reportthis指向:

class Report {
    public:
        template < typename T >
        Report& operator <<( const T &data ) { ... }
};

或者作为友元函数,其中没有this,而是明确提及和命名Report

class Report {
    public:
        template < typename T >
        friend Report& operator <<( Report &r, const T &data ) { ... }
};

或者作为独立函数,如果Report有足够的公共方法来完成这项工作。

template < typename T >
Report& operator <<( Report &r, const T &data ) { ... }

要将其实现为成员函数并能够连接多个流,您只需在函数末尾返回对对象的引用,例如:

class Report {
public:
    // ...
    Report& operator<<(const char *text) {
        // write text to file...
        return *this;
    }
};

下面是一个非常简单的示例,它打印到标准输出而不是文件: http://codepad.org/S8QeJZ1x

如果您只想确保文件在使用后关闭,则不需要类Reportstd::ofstream 的析构函数已经关闭了它的文件。

到目前为止,其他试图定义模板化operator<<的答案有一个问题:它们不适用于包括std::endl在内的某些操纵器,因为它实际上是一个函数模板,编译器无法弄清楚它应该使用哪种实例化。

如果类Report的目的是在正在通过的数据中添加或修改某些内容,那么通过定义自己的 streambuf 可以更容易、更正确地完成此操作。 James Kanze的这篇文章描述了如何做到这一点。