c++ - 设置 boost::log 严重性过滤器来调用函数,而不是使用常量严重性

标签 c++ logging boost

我创建了一个使用 boost::log 的 Logger 类。该类跟踪当前严重性级别集。将严重性过滤器添加到接收器时,我希望过滤器根据当前严重性进行过滤,而不是添加接收器时的过滤器。

class Logger
{
public:
    typedef /* ... */ severity_level;
    //(...)
    static severity_level currentSeverityLevel() {return severity_level_var;}
private:
    severity_level severity_level_var;
    //(...)
};

添加过滤器时,如下所示,仅在设置过滤器时调用 currentSeverityLevel()。

console_sink->set_filter( severity >= Logger::currentSeverityLevel());

我想要这样的过滤器,但实际上每次都会调用该函数。根据文档,可以定义一个函数用作过滤器(见下文),但它需要使用 boost::phoenix。当然必须有更简单的方法吗?类似 lambda 表达式的东西? set_filter 函数设置了一个过滤器类型,它又设置了一个 boost light_function,它是“Boost.Function 的轻量级替代品”。那么,似乎可以直接设置类似于 Lambda 表达式/Boost.Function 的内容

bool my_filter(logging::value_ref< severity_level, tag::severity > const& level,
               logging::value_ref< std::string, tag::tag_attr > const& tag)
{
    return level >= warning || tag == "IMPORTANT_MESSAGE";
}

void init()
{
    // ...
    namespace phoenix = boost::phoenix;
    sink->set_filter(phoenix::bind(&my_filter, severity.or_none(), tag_attr.or_none()));
    // ...
}

最佳答案

为了在每个日志记录上调用 Logger::currentSeverityLevel,您必须将其转换为 Boost.Phoenix 表达式,以便对其进行惰性计算。最简单的方法是使用 phoenix::bind

console_sink->set_filter(
    severity >= phoenix::bind(&Logger::currentSeverityLevel)
);

如果按照 here 所述为您的函数创建惰性包装器,您可以获得更清晰的语法。 .

BOOST_PHOENIX_ADAPT_FUNCTION_NULLARY(
    Logger::severity_level,
    lazyCurrentSeverityLevel,
    Logger::currentSeverityLevel
);

console_sink->set_filter(severity >= lazyCurrentSeverityLevel());

此外,您还可以在 Boost.Phoenix 表达式中保存对外部对象的引用。

console_sink->set_filter(severity >= phoenix::ref(severity_level_var));

此过滤器将在每次调用时读取 severity_level_var 值。但是,这种方法使确保对 severity_level_var 变量的线程安全操作变得更加困难。这不是在您的 currentSeverityLevel 实现中完成的,因此我假设线程安全不是必需的,就是通过其他方式实现的。

关于c++ - 设置 boost::log 严重性过滤器来调用函数,而不是使用常量严重性,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/35049633/

相关文章:

c++ - 传递回调成员函数的高效方法

Android NDK C++ 'wstring' 支持

sql-server - 如何在 SSIS 中创建日志文件

c++ - 使用外部变量时出现多重定义错误

java - 用于多个应用程序的多个 log4j xml 文件

android - 如何在Android中查看Gstreamer日志?

C++17 resumable/await:它也可以与 boost::future 一起使用吗?

c++ - 除非已经转义,否则如何替换所有出现的字符串或字符?

c++ - 位图到 mat/2d 数组

c++ - 初始化C++基类,而无需两次键入整个基类类型