【问题标题】:Cannot resolve ILogger<T> Simple Injector ASP.NET Core 2.0无法解析 ILogger<T> 简单注入器 ASP.NET Core 2.0
【发布时间】:2019-03-27 00:08:31
【问题描述】:

我们在 ASP.NET Core 应用程序中使用 Simple Injector。最近我们决定使用 Serilog 进行日志记录。

documentation 所述,在 Program.cs 中完成了配置。然后,为了让 Simple Injector 能够解析ILoggerFactory,我做了这样的事情:

public void Configure(IApplicationBuilder app, IHostingEnvironment env)
{
    if (env.IsDevelopment())
    {
        app.UseDeveloperExceptionPage();
    }
    app.UseMvc();

    // As per my understanding, we force SimpleInjector to use 
    // ILoggerFactory registration from IApplicationBuilder
    container.CrossWire<ILoggerFactory>(app)
}

之后,ILoggerFactory 可以在我们使用的任何地方注入。它工作正常(创建新实例ILogger&lt;T&gt;)。

但这很烦人,每次需要时都从工厂创建实例。 最好直接在依赖类中获取ILogger&lt;T&gt;,而不是ILoggerFactory

那么,我如何注册ILogger&lt;T&gt; 以获取实例,以防出现如下情况?

public class HelloWorldController : Controller
{
    public HelloWorldController(ILogger<HelloWorldController> logger)
    {
         // ...
    }
}

【问题讨论】:

    标签: c# asp.net-core simple-injector serilog


    【解决方案1】:

    虽然 Alsami 的答案可行,但请改用以下注册:

    container.RegisterConditional(
        typeof(ILogger),
        c => typeof(Logger<>).MakeGenericType(c.Consumer.ImplementationType),
        Lifestyle.Singleton,
        c => true);
    
    // This next call is not required if you are already calling AutoCrossWireAspNetComponents
    container.CrossWire<ILoggerFactory>(app);
    

    这个确切的例子显示在documentation

    此注册允许将Logger&lt;T&gt; 注入非通用 ILogger 构造函数参数,其中Logger&lt;T&gt;T 成为记录器注入的类型。换句话说,当HelloWorldController 依赖于ILogger 时,它将被注入Logger&lt;HelloWorldController&gt;。这意味着您可以将 HelloWorldController 简化为以下内容:

    public class HelloWorldController : Controller
    {
        public HelloWorldController(ILogger logger)
        {
             // ...
        }
    }
    

    通过让您的应用程序组件依赖于ILogger 而不是ILogger&lt;T&gt; 您:

    • 简化您的应用程序代码
    • 简化您的单元测试
    • 消除出现意外错误的可能性,因为不可能注入错误的记录器。

    【讨论】:

    • 将记录器注册为单例是一个坏主意,因为记录器的创建很重,并且您必须传递一个类型,即使有目的并且应该传递。 stackoverflow.com/a/8485625/5397642
    • Logger&lt;T&gt; 是线程安全的,可以安全地注册为单例。事实上,ASP.NET Core 本身将其注册为单例。
    【解决方案2】:

    您还必须将记录器本身注册为泛型类型。 我不知道简单的注入器,但这应该是正确的语法。

    container.Register(typeof(ILogger<>), typeof(Logger<>), Lifestyle.Singleton);
    

    【讨论】:

    • 天哪!再简单不过了。你是救命稻草!按预期工作。谢谢!
    猜你喜欢
    • 2017-05-05
    • 1970-01-01
    • 2016-11-10
    • 1970-01-01
    • 2019-02-19
    • 1970-01-01
    • 2018-10-08
    • 1970-01-01
    相关资源
    最近更新 更多