设置boost :: log severity过滤器以调用函数,而不是使用常量严重性

时间:2016-01-27 22:49:07

标签: 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()));
    // ...
}

1 个答案:

答案 0 :(得分:1)

为了在每个日志记录上调用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实现中完成的,因此我认为不需要或通过其他方式实现线程安全。