插入输出后,ostringstream为空

时间:2018-05-26 09:53:48

标签: c++ stl ostringstream

我有一个奇怪的问题,即使我输入一个输出,ostringstream也是空的。 这是我的代码:

//logger.h

#include <string>
#include <iostream>
#include <sstream>
using std::string;


class BasicLogger {
public:
    BasicLogger(const string name);
    BasicLogger(const BasicLogger& basicLogger);
    ~BasicLogger();

    template<class T>
    BasicLogger& operator<<(const T &msg){
        std::cout << "msg is: " << msg << std::endl;
        mBuf << msg;
        std::cout << "mBuf is: " << mBuf.str() << std::endl;
        return *this;
    }


private:
    string mName;
    std::ostringstream mBuf;
};


class Logger {
public:
    Logger();
    ~Logger();
    BasicLogger info();
    BasicLogger error();

private:
    BasicLogger mInfoLogger;
    BasicLogger mErrorLogger;

};
//logger.cpp

#include "logger.h"

BasicLogger::BasicLogger(const string name):
    mName(name) { }

BasicLogger::BasicLogger(const BasicLogger& otherLogger) {
    this->mName = otherLogger.mName;
    this->mBuf << otherLogger.mBuf.rdbuf();
}

BasicLogger::~BasicLogger() { }

Logger::Logger(): 
    mInfoLogger("[INFO]"),
    mErrorLogger("[ERROR]") {}

Logger::~Logger() {};

BasicLogger Logger::info() {
    return mInfoLogger;
}

BasicLogger Logger::error() {
    return mErrorLogger;
}
//main.cpp

#include "logger.h"

int main() {
    Logger logger;
    logger.info() << "Hellooo";
}

输出

msg is: Hellooo
mBuf is: 

1 个答案:

答案 0 :(得分:1)

虽然@BoPersson已经为您提供了解决方案,但我想解释一下这里发生了什么,以及为什么您的输出为空,即使您在显示其内容之前在行上添加ostringstream。< / p>

我认为当您尝试按值返回mInfoLogger时,编译器抱怨无法返回它,因为复制构造函数已被删除。因为ostringstream成员而被删除的原因是不可复制的。

因此,您提供了自定义复制构造函数,并尝试以这种方式复制流:

this->mBuf << otherLogger.mBuf.rdbuf();

只有这不会复制任何内容,因为rdbuf为空,而是设置failbit

  

failbit

     

basic_ostream :: operator&lt;&lt;的streambuf重载如果功能   不插入任何字符。   http://en.cppreference.com/w/cpp/io/ios_base/iostate

如果您要按照以下方式检查邮件插入:

if (!(mBuf << msg))
    std::cout << "Not Inserted" << std::endl;

您会看到Not Inserted消息已打印出来。有许多方法可以打印所需的信息。例如,确保您的rdbuf缓冲区不为空,或者在重用failbit之前重置mBuf,例如mBuf.clear();,或者只是将引用返回{{1} (和mInfoLogger)。然后你可以安全地摆脱你的拷贝构造函数。