- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我想在代码中为记录器创建一个附加程序,但同时我想在属性文件中进行配置。我正在以这种方式创建 appender:
class Foo {
private:
LoggerPtr logger;
public:
Foo(int id) {
logger = LoggerPtr(Logger::getLogger("foo." + std::to_string(id)));
RollingFileAppender* appender = new RollingFileAppender();
appender->setAppend(true);
appender->setMaxFileSize("1MB");
appender->setMaxBackupIndex(1);
appender->setFile("foo." + std::to_string(id) + ".log");
PatternLayout* layOut = new PatternLayout();
layOut->setConversionPattern(
"%d{yyyy-MM-dd HH:mm:ss} %-5p %c{1}:%L - %m%n");
appender->setLayout(LayoutPtr(layOut));
appender->setName("fooAppender");
Pool p;
appender->activateOptions(p);
logger->addAppender(AppenderPtr(appender));
logger->setAdditivity(false);
}
const LoggerPtr& getLogger() const {
return logger;
}
};
这里的目标是为 Foo
类的每个实例创建不同的文件,并且由于数字是动态的,我无法在属性文件中创建记录器(也许我可以从 0 到 2 *1024*1024 但这太疯狂了)。为了实现这个目标,我需要几个附加程序,但是我希望在属性文件中有类似的东西:
log4j.appender.foo=org.apache.log4j.RollingFileAppender
log4j.appender.foo.MaxFileSize=50MB
log4j.appender.foo.MaxBackupIndex=1
log4j.appender.foo.layout=org.apache.log4j.PatternLayout
log4j.appender.foo.layout.ConversionPattern=%p %t %c - %m%n
是否可以让自定义 appender 继续从属性文件中读取配置?
最佳答案
我自己回答。你需要什么:你需要创建一个新的自定义附加程序并将其设置为第一级记录器中的附加程序,而不是在代码中你可以克隆附加程序。作为奖励,在您的自定义附加程序中,您可以管理延迟的文件创建,因此如果没有日志,则不再有空文件;)这里是代码:
class MyLogger {
private:
MyLogger() {
}
public:
static void cloneParentLogger(const std::string& parentName,
const std::string& parentAppenderName, LoggerPtr& child,
const std::string& childName) {
std::string newName = parentName + "." + childName;
LoggerPtr loggerMst = LoggerPtr(Logger::getLogger(parentName));
Appender* app = loggerMst->getAppender(parentAppenderName);
app->close();
AppenderPtr newAppender = app->getClass().newInstance();
if (newAppender->instanceof(FileAppender::getStaticClass())) {
Appender* p = newAppender;
reinterpret_cast<FileAppender*>(p)->setFile(newName + ".log");
}
child = LoggerPtr(Logger::getLogger(newName));
newAppender->setName(childName);
newAppender->setLayout(app->getLayout());
Pool p;
newAppender->activateOptions(p);
child->addAppender(newAppender);
child->setAdditivity(false);
}
};
class Foo {
private:
LoggerPtr logger;
public:
Foo(int id) {
MyLogger::cloneParentLogger("foo", "R1", logger, std::to_string(id));
}
const LoggerPtr& getLogger() const {
return logger;
}
};
属性文件
# Set root logger level to DEBUG and its only appender to A1.
log4j.rootLogger=DEBUG
log4j.logger.foo=DEBUG, R1
log4j.appender.R1=DelayedRollingFileAppender
log4j.appender.R1.MaxFileSize=50MB
log4j.appender.R1.MaxBackupIndex=1
log4j.appender.R1.layout=org.apache.log4j.PatternLayout
log4j.appender.R1.layout.ConversionPattern=%p %t %c - %m%n
具有延迟文件创建的自定义附加器:
#ifndef DELAYEDROLLINGFILEAPPENDER_H_
#define DELAYEDROLLINGFILEAPPENDER_H_
#include <log4cxx/rollingfileappender.h>
#include <log4cxx/helpers/object.h>
#include <log4cxx/helpers/pool.h>
#include <log4cxx/helpers/classregistration.h>
#include <log4cxx/logstring.h>
namespace log4cxx {
class DelayedRollingFileAppender: public RollingFileAppender {
private:
bool firstAppend;
LogString file;
bool app;
bool bufferedIo;
size_t bufferSize;
public:
//DECLARE_LOG4CXX_OBJECT(DelayedRollingFileAppender)
class ClazzDelayedRollingFileAppender: public helpers::Class {
public:
ClazzDelayedRollingFileAppender() :
helpers::Class() {
}
virtual ~ClazzDelayedRollingFileAppender() {
}
virtual LogString getName() const {
return LOG4CXX_STR("DelayedRollingFileAppender");
}
virtual helpers::ObjectPtr newInstance() const {
return new DelayedRollingFileAppender();
}
};
virtual const helpers::Class& getClass() const;
static const helpers::Class& getStaticClass();
static const helpers::ClassRegistration& registerClass();
DelayedRollingFileAppender();
virtual ~DelayedRollingFileAppender();
virtual void append(const spi::LoggingEventPtr& event,
helpers::Pool& pool1);
virtual void setFile(const LogString& file, bool append, bool bufferedIO,
size_t bufferSize, helpers::Pool& p);
};
} /* namespace log4cxx */
#endif /* DELAYEDROLLINGFILEAPPENDER_H_ */
Cpp文件:
#include "DelayedRollingFileAppender.h"
#include <log4cxx/writerappender.h>
#include <log4cxx/helpers/fileoutputstream.h>
#include <log4cxx/helpers/bufferedwriter.h>
namespace log4cxx {
using namespace helpers;
//IMPLEMENT_LOG4CXX_OBJECT(DelayedRollingFileAppender)
const ::log4cxx::helpers::Class& DelayedRollingFileAppender::getClass() const {
return getStaticClass();
}
const ::log4cxx::helpers::Class& DelayedRollingFileAppender::getStaticClass() {
static ClazzDelayedRollingFileAppender theClass;
return theClass;
}
const helpers::ClassRegistration& DelayedRollingFileAppender::registerClass() {
static helpers::ClassRegistration classReg(
DelayedRollingFileAppender::getStaticClass);
return classReg;
}
namespace classes {
const ::log4cxx::helpers::ClassRegistration& DelayedRollingFileAppenderRegistration =
DelayedRollingFileAppender::registerClass();
}
DelayedRollingFileAppender::DelayedRollingFileAppender() {
firstAppend = true;
this->app = false;
this->bufferSize = 0;
this->bufferedIo = false;
}
DelayedRollingFileAppender::~DelayedRollingFileAppender() {
}
void DelayedRollingFileAppender::append(const spi::LoggingEventPtr& event,
helpers::Pool& pool1) {
if (firstAppend) {
firstAppend = false;
FileAppender::setFile(file, app, bufferedIo, bufferSize, pool1);
}
WriterAppender::append(event, pool1);
}
void DelayedRollingFileAppender::setFile(const LogString& file, bool append,
bool bufferedIO, size_t bufferSize, helpers::Pool& p) {
this->file = file;
this->app = append;
this->bufferSize = bufferSize;
this->bufferedIo = bufferedIO;
OutputStreamPtr outStream;
try {
outStream = new FileOutputStream("/dev/null", false);
} catch (IOException& ex) {
throw;
}
WriterPtr newWriter(createWriter(outStream));
if (bufferedIo) {
newWriter = new BufferedWriter(newWriter, bufferSize);
}
setWriter(newWriter);
}
} /* namespace log4cxx */
关于c++ - 使用 log4cxx 配置自定义和动态附加程序,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/51688119/
我想知道是否有办法只在编译时而不是在链接时将一些标志传递给编译器 (g++)。让我用一个例子来解释:我想这样构建我的程序: g++ -c source1.cpp -o source1.o g++ -c
我们在使用 Visual Studio 2012 在 Windows 7 64Bit 下使用 CMake (v2.8.12) 编译项目时遇到问题。CMake 给我们以下错误。我们已经尝试使用管理员权限
我是 Linux 的新用户,正在尝试下载 Avogadro 1.2.0。 我在使用命令 cmake ../ 时遇到问题,当我这样做时 -- The C compiler identification
我使用的是 Julia 1.1.1 和 Cxx 的 0.3.2 版。我发现当我从我定义的模块中调用 icxx""时出现错误,但在 REPL 中调用它时却没有。在我的 REPL 中: using Cxx
./cmake-3.4.1/bin/cmake .. -- The CXX compiler identification is unknown CMake Error at CMakeLists.t
我的代码库中有大量文件。我正在尝试使用具有一个文件 a.h. 的其他库来编译我的代码库。如果我在我的代码库中包含 say a.h 文件,我会遇到编译问题,该文件已经定义了一些具有与 a.h 中定义的枚
这个问题在这里已经有了答案: 关闭 10 年前。 Possible Duplicate: Where does the value of CXX in a makefile come from? 我
我正在尝试使用 Cmake 将外部库安装到我的 C++ 项目中。我希望使用该库生成 Xcode 项目。在我的终端中,我从构建目录运行以下命令: cmake -G Xcode .. 这给了我以下错误:
我正在尝试在ubuntu 16.10上交叉编译ndn-cxx以在arduino yun上使用它。我正在关注steps。 当我尝试执行./waf命令时,出现以下错误: [ 30/141] Com
来自CXX Driver Quickstart guide我完全无法在 Ubuntu 或 CentOS VM 上完成新驱动程序的编译。 我已经按照这封信中的步骤执行了好几次,但我不确定哪里出了问题。
我有一个类,它有一个 bsoncxx::document::view view 作为 private 属性,还有一个从 Mongo 数据库获取文档并保存的方法结果在本地 bsoncxx::docume
这个问题在这里已经有了答案: gitignore does not ignore folder (7 个答案) 关闭 10 个月前。 我的 android 项目中有这个 .gitignore 文件,
我正在尝试在 Ubuntu14.4 上编译 mongo-cxx-driver-r3.1.1 。我已经按照说明从源代码安装了 mongo-c-driver-1.6.3。 pkg-config --cfl
我尝试让我的库目标依赖于 c++14。我怎样才能实现链接到我的库目标的目标也依赖于 c++14? 我尝试使用: add_library(lib SHARED ${FILES}) target_comp
帮助!我在提交我的项目使用 dbus-cxx 时可能犯了一个巨大的错误。该库似乎已被其作者遗弃,并且邮件列表对新成员不开放。这是我的问题: 没有关于使用 dbux-cxx 发送或接收包含非 POD 数
我使用 dbus-cxx通过 dbus 进行方法调用。现在有返回多个参数的方法,我不知道如何接收第一个参数以外的任何参数。 我初始化代理方法: DBus::MethodProxy& info_prox
CXX 测试框架的有效性如何,假设您正在围绕您编写的代码编写单元测试用例。代码中的任何错误也可能会转化为单元测试代码中的错误吗?这不是两个负数组成一个正数吗? 此外,花在 CXX 上的时间和精力至少等
在 cmake 已经预编译失败后,在 99% 编译时链接 CXX 可执行世界服务器。 [ 99%] Linking CXX executable worldserver /usr/bin/ld: ..
代码片段: target_test : test.cc $(CXX) $(CPPFLAGS) $(CFLAGS) test.cc 我知道 CXX 是一个变量(包含要调用的编译器命令),但我想知
我正在使用以下代码查询一个集合: bsoncxx::stdx::optional query_result = collection.find_one(bsoncxx::builder::stream
我是一名优秀的程序员,十分优秀!