带有DI Simple Injector的log4net

时间:2014-08-04 05:28:17

标签: c# .net asp.net-mvc dependency-injection simple-injector

我正在尝试使用Simple Injector(+ integration MVC)v 2.5.2。对于MVC 4应用程序,我也需要跟踪/记录性能(执行)(通过log4net模块)。当前实现(在运行时)在指定路径中创建log4net文件,但没有写入任何文本行(当我调试它时,一切都没有错误到 _logger.Info(“message”))。

有没有人尝试使用Simple Injector DI for 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);

    }
}

在Global.asax中

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

我做错了什么?在我看来注射方式的问题。 感谢您的任何建议

1 个答案:

答案 0 :(得分:14)

在配置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>实施。

请注意,虽然IMO可以更好地防止应用程序代码依赖第三方库抽象。我认为define you own logger abstraction要好得多。

我对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配置错误时,您的应用程序会直接停止并显示明确的异常。