需求:想要实现一个Logger可以以以下方式使用:
Logger log("2013-03-05.log");
log << "This is a logger test" << endl;
log << "value of Temp = " << temp << endl;
log << "Array[" << index << "] = " << array[index] << endl;
实现有一点类似cout。
下面是我定义Logger类:
class Logger
{
public:
Logger(const QString& logfilename);
void sync(void); // 将m_logStr写入log文件
Logger& operator << (QString str) { m_logStr += str; return (*this); }
Logger& operator << (int t) { m_logStr += t; return (*this); }
// Other operators, QChar, const char*, unsigned int, etc
// ...
Logger& operator << ( Logger& (*op) (Logger&) ) { return (*op)(*this); }
private:
QString m_logStr;
}
static Logger& endl (Logger &log) { log.sync(); return log; }
如何用<<连接大家应该都很容易理解,关键技术点在这个endl。
其中最后一个 operator << 重载告诉编译器:如果<<后面是一个(Logger&)(*op)(Logger&)的函数指针,那么就这么调用它:(*op)(*this)
然后在全局中实现了Logger& endl(Logger&)的函数。当log << endl 时,就会调用这个endl函数并执行。
就这样,我何可以定义很多类似的流符号了。
那个可不可以这样呢?
log << size(1024) << "Set size ..." << endl;
我们需要可以像上面size()可以带参的,怎么实现?
struct size{
size(int d): value(d){}
int data;
};
class Logger
{
public:
// Other ...
Logger& operator << ( size v ) { m_nSize = v.data; return (*this); }
private:
int m_nSize;
}
定义结构体size,并指定其构造函数size(int value)。所以"log << size(1024)"中的"size(1024)"其实是在构造一个size结构类型的变量,而1024就被存到了size::nSize中。
这就是我今天的收获!谢谢观看~~