我想在ASP.NET Core中实现依赖注入(DI)。因此,在将此代码添加到ConfigureServices方法后,这两种方法都可以工作。

ASP.NET Core中的services.AddTransient和service.AddScoped方法之间有什么区别?

public void ConfigureServices(IServiceCollection services)
{
    // Add framework services.

    // Add application services.
    services.AddTransient<IEmailSender, AuthMessageSender>();
    services.AddScoped<IEmailSender, AuthMessageSender>();
}

当前回答

Singleton是应用程序生命周期中的单个实例领域作用域是作用域期间的单个实例请求,这意味着ASP.NET中的每个HTTP请求。瞬态是每个代码请求的单个实例。

通常,代码请求应该通过构造函数参数进行,如

public MyConsumingClass(IDependency dependency)

我想在@akazemis的回答中指出,DI上下文中的“服务”并不意味着RESTful服务;服务是提供功能的依赖项的实现。

其他回答

在.NET的依赖注入中,有三个主要的生命周期:

在整个应用程序中创建单个实例的Singleton。它第一次创建实例,并在所有调用中重用相同的对象。

作用域内的每个请求创建一次作用域生存期服务。它相当于当前范围中的单例。例如,在MVC中,它为每个HTTP请求创建一个实例,但在同一web请求中的其他调用中使用相同的实例。

每次请求时都会创建短暂的生命周期服务。这种生命周期最适合于轻量级、无状态服务。

在这里,您可以找到和示例来了解不同之处:

ASP.NET 5 MVC6依赖注入,分6个步骤进行(由于链接无效,导致web存档链接)

依赖项注入就绪的ASP.NET:ASP.NET 5

这是官方文件的链接:

ASP.NET核心中的依赖注入

当必须注入同一类型的多个对象时,ASP.NET MVC核心DI(依赖注入)中的瞬时、作用域和单例定义对象创建过程。如果你是依赖注入的新手,你可以看到这个DI-IoC视频。

您可以看到下面的控制器代码,其中我在构造函数中请求了两个“IDal”实例。Transient、Scoped和Singleton定义了同一个实例是否将以“_dal”和“_dal1”或不同的方式注入。

public class CustomerController : Controller
{
    IDal dal = null;

    public CustomerController(IDal _dal,
                              IDal _dal1)
    {
        dal = _dal;
        // DI of MVC core
        // inversion of control
    }
}

瞬态:在瞬态中,将在单个请求和响应中注入新的对象实例。下面是我显示GUID值的快照图像。

作用域:在作用域中,相同的对象实例将被注入到单个请求和响应中。

Singleton:在Singleton中,将在所有请求和响应中注入相同的对象。在这种情况下,将创建对象的一个全局实例。

下面是一个简单的图表,直观地解释了上述基本原理。

上图是我在孟买参加ASP.NET MVC培训时,SBSS团队绘制的。非常感谢SBSS团队创建了上述图像。

使用哪一个

转瞬即逝的

因为它们每次创建时都会使用更多的内存和资源,并且会对性能产生负面影响对于状态很少或没有状态的轻量级服务,请使用此选项。

范围内的

当您想要维护请求中的状态时,更好的选项。

辛格尔顿

这些服务中的内存泄漏会随着时间的推移而增加。也可以节省内存,因为它们一旦在任何地方重复使用就会被创建。

在需要维护应用程序范围状态的地方使用Singleton。应用程序配置或参数、日志服务、数据缓存是可以使用单例的一些示例。

将具有不同生存期的服务注入另一个

切勿将作用域和瞬态服务注入Singleton服务。(这有效地将瞬时或作用域服务转换为单例。)从不将瞬态服务注入作用域服务(这会将瞬态服务转换为作用域。)

添加单体()

AddSingleton()在首次请求时创建服务的单个实例,并在需要该服务的所有地方重用该实例。

添加作用域()

在作用域服务中,对于每个HTTP请求,我们都会得到一个新实例。然而,在同一HTTP请求中,如果在多个位置(如视图和控制器中)需要服务,则为该HTTP请求的整个范围提供相同的实例。但是,每个新的HTTP请求都将获得服务的新实例。

添加过渡()

对于瞬时服务,无论服务实例是在同一HTTP请求的范围内还是跨不同HTTP请求,每次请求服务实例时都会提供一个新实例。

通过DbContext使用EntityFramework/Core可能是生命周期的最佳说明。

建议将DbContext和与DbContext交互的存储库连接到Scoped生存期,因为DbContext显然是一个有状态的构造。所以你不想使用Singleton,因为你最终会遇到各种并发问题。您不想使用Transient,因为DbContext不是线程安全的。记住,Transient适用于处理无状态对象/类的用例。

而且,由于大多数存储库都是由控制器调用的,所以使用作用域生存期确实很有意义。可以想象,作为事务的一部分,DbContext可以在单个操作方法期间多次调用。

本文没有直接讨论这些生存期,但对为什么Scoped生存期最适合DbContext给出了很好的解释。

https://mehdi.me/ambient-dbcontext-in-ef6/?msclkid=00251b05d01411ec8d85d232374f26d5