log4net与DI简单注入器
本文关键字:注入器 简单 DI log4net | 更新日期: 2023-09-27 18:16:09
我试图使用简单注入器(+集成MVC) v 2.5.2。对于MVC 4应用程序,我需要跟踪/日志性能(执行)以及(通过log4net模块)。当前的实现(在运行时)在指定的路径中创建log4net文件,但没有向其中写入任何文本行(当我调试它时,一切都没有错误地到_logger.Info("message")的末尾)。
是否有人试图使用简单注入器DI为log4net?
我注册log4net模块的方式是:
public static class LoggingModule
{
public static void RegisterServices(Container container)
{
string log4NetConfigFile = Path.Combine(AppDomain.CurrentDomain.BaseDirectory, "log4net.config");
log4net.Config.XmlConfigurator.ConfigureAndWatch(new FileInfo(log4NetConfigFile));
var logger = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType);
container.RegisterSingle(logger);
}
}
LoggingModule.RegisterServices(container);
Log4net配置文件看起来像这样(我不认为有任何问题):
<log4net>
<appender name="PerformanceLogFile" type="log4net.Appender.RollingFileAppender" >
<param name="File" value="..'Performance'PricingManager" />
<param name="AppendToFile" value="true" />
<param name="RollingStyle" value="Date" />
<param name="DatePattern" value="_yyyy-MM-dd.lo'g" />
<param name="MaxSizeRollBackups" value="10" />
<param name="StaticLogFileName" value="false" />
<layout type="log4net.Layout.PatternLayout">
<param name="ConversionPattern" value="%m%n"/>
</layout>
<filter type="log4net.Filter.LevelRangeFilter">
<param name="LevelMin" value="INFO" />
<param name="LevelMax" value="FATAL" />
</filter>
</appender>
<logger name="SoftTech.PricingEngine.PricingService.PerformanceStatisticLoggerWithAPI">
<level value="ALL" />
<appender-ref ref="PerformanceLogFile" />
</logger>
</log4net>
打开/关闭日志记录,我一直在使用:
private static void RegisterIPerformanceStatisticLogger(Container container)
{
if (ShouldLogPerformance())
{
container.Register<IPerformanceStatisticLogger, PerformanceStatisticLogger>(WebLifestyle);
}
else
{
// do nothing
container.Register<IPerformanceStatisticLogger, DisabledPerformanceStatisticLogger>(WebLifestyle);
}
}
和PerformanceStatisticLogger或DisablePerformanceStatisticLogger设置
IsLoggingEnabled = true; // | false
IsAPITraceEnabled = false; // | true
我做错了什么?在我看来是注射方式的问题。谢谢你的建议
在配置log4net时,Simple Injector没有什么特别之处。例如,您可以按照如下方式注册ILog
:
container.RegisterSingleton<ILog>(LogManager.GetLogger(typeof(object)));
显然,这为整个应用程序注册了一个记录器。如果您希望为每个类注入不同的记录器,则必须定义自己的适配器实现:
public sealed class Log4NetAdapter<T> : LogImpl
{
public Log4NetAdapter() : base(LogManager.GetLogger(typeof(T)).Logger) { }
}
您可以按照以下方式注册此适配器:
container.RegisterConditional(typeof(ILog),
c => typeof(Log4NetAdapter<>).MakeGenericType(c.Consumer.ImplementationType),
Lifestyle.Singleton,
c => true);
这确保每个消费者获得自己的Log4NetAdapter<T>
实现。
请注意,在我看来,防止应用程序代码依赖于第三方库抽象要好得多。我认为最好定义自己的日志抽象。
我对log4net的经验是,当log4net不记录任何东西时,您的log4net配置有问题。我讨厌log4net的一点是,当您配置错误时,它永远不会抛出任何异常。它只是吞噬并继续,这使得使用log4net比它应该的要痛苦得多。
可以帮助的一件事是在启动时连接到log4net的LogLog.LogReceived
事件。这允许您检查是否有任何错误。例如,这是我在以前的项目中使用的代码:
LogLog.LogReceived += (s, e) =>
{
if (e.LogLog.Prefix.Contains("ERROR"))
{
throw new ConfigurationErrorsException(e.LogLog.Message,
e.LogLog.Exception);
}
};
这可以确保当log4net配置错误时,您的应用程序直接停止并出现明显的异常。