我创建了一个使用 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/