如何将模板可变参数存储到std :: ofstream中?

时间:2018-02-16 16:58:31

标签: c++ c++11 templates logging variadic-templates

我正在c ++中实现一个小Log

class Log
    {
    public:
        enum LogLevel
        {
            TRACE,
            DEBUG,
            INFO,
            WARNING,
            ERROR,
            FATAL
        };

        Log(): m_log(DEBUG) {} ;
        ~Log() {};

        int SetLogFilePath(const std::string& p_directory, const std::string& p_prefix);

        void SetLogLevel(const LogLevel p_logLvl);

        LogLevel& GetLogLevel();

        template<typename... Args>
        int Logging(const std::string& p_logLevel, const std::string& p_srcFile, const std::string& p_function, const int& p_line,
            const Args&& ... p_args);

    private:
        LogLevel m_log;
        std::string m_logFilePath;
    };

这是我的Log类,我将目录和前缀存储到std::string m_logFilePath

template <typename ... Args>
    int Log::Logging(const std::string& p_logLevel, const std::string& p_srcFile, const std::string& p_function, const int& p_line,
        const Args&&... p_args)
    {
        std::ofstream log;
        log.open(m_logFilePath.c_str(), std::ios::out);
        if(!log.good())
        {
            std::cout << "Couldn't create file : " << m_logFilePath.c_str() << std::endl;
            return 0;
        }
        switch(p_logLevel)
        {
        case "TRACE":
            log << "(TRACE) ";
            break;

        case "DEBUG":
            log << "(DEBUG) ";
            break;

        case "INFO":
            log << "(INFO) ";
            break;

        case "WARNING":
            log << "(WARNING) ";
            break;

        case "ERROR":
            log << "(ERROR) ";
            break;

        case "FATAL":
            log << "(FATAL) ";
            break;

        default: break;
        }
        log << "Log Created at (Local Time): " << std::put_time(std::localtime(std::chrono::system_clock::to_time_t(std::chrono::system_clock::now())), "%c") << " ";
        log << "File Name: " << p_srcFile.c_str() << " " << "Function Name: " << p_function.c_str() << " " << "Line: " << p_line << " " << std::forward<Args>(p_args) << std::endl;
        log.close();
        return 1;
    }

以上是int Logging(...)函数的实现。

基本上,我存储了日志记录级别,当前时间,源文件路径,函数名称,代码行和一些可变参数。

我一直在寻找一种方法来将多个可变参数存储到我的std::ostream log中,但找不到任何答案。

请帮忙!

1 个答案:

答案 0 :(得分:1)

如果我理解正确,你的问题是你不知道如何管理(在这种情况下:发送到输出流)一个可变参数列表(基于可变参数模板)。

我提出了一个在(未使用的)数组初始化的上下文中使用逗号运算符的强大功能的使用示例

http://www\.\.com[^<]+

以下是一个简化但完整的示例

using unused = int[];

int  cnt { -1 };

(void)unused { 0, (oss << ", extra val "
                       << ++cnt << " [" << args << ']', 0)... };