我创建了一个使用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()));
// ...
}
答案 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
实现中完成的,因此我认为不需要或通过其他方式实现线程安全。