ASP.NET 核心 DI 与简单注入器共享 DbContext

ASP.NET Core DI with Simple Injector sharing DbContext

我正在建立一个 ASP.NET 核心项目,并根据我在 Tripod 项目上的工作遵循 CQRS 模式。我遵循了 Simple Injector integration guide 但有一点有点困惑......我想继续使用 UseInMemoryDatabase 选项进行测试并且只能找到使用 Core DI [=17= 的示例]方法。

我的 DbContext 实现了几个接口:

public class EntityDbContext : DbContext,
    IUnitOfWork,
    IReadEntities,
    IWriteEntities
{
    // code here
}

我在 Startup.ConfigureServices 方法中这样做:

services.AddDbContext<EntityDbContext>(options => options.UseInMemoryDatabase("Snoogans"));

并在 Startup.Configure 方法中遵循 SI 集成指南:

container.Register<IUnitOfWork, xxxx>();
container.Register<IReadEntities, xxxx>();
container.Register<IWriteEntities, xxxx>();

我是否能够通过交叉线获得核心注册以插入到每个目标的目标中,或者我应该直接在 SI 中注册上下文?

== 我正在研究 Tripod 的原始概念:

var contextRegistration =
    lifestyle.CreateRegistration<EntityDbContext, EntityDbContext>(container);
container.AddRegistration(typeof(EntityDbContext), contextRegistration);
container.AddRegistration(typeof(IUnitOfWork), contextRegistration);
container.AddRegistration(typeof(IWriteEntities), contextRegistration);
container.AddRegistration(typeof(IReadEntities), contextRegistration);

尝试使用 SI 做所有事情,但不确定我如何获得 3 个接口的注册:

container.Register<EntityDbContext>(() =>
{
    var optionsBuilder =
        new DbContextOptionsBuilder<EntityDbContext>().UseInMemoryDatabase("Snoogans");
    return new EntityDbContext(optionsBuilder.Options);
});
container.AddRegistration<IUnitOfWork>(xxxx);
container.AddRegistration<IReadEntities>(xxxx);
container.AddRegistration<IWriteEntities>(xxxx);

Am I able to get the Core registration over crosswire to plugin to the target for each of them or should I just be registering the context directly in SI?

两种方案都可行。您可以选择将 DbContext 直接注册到 Simple Injector 中。这通常是最明显的选择,因为您的 DbContext 是一个应用程序组件。应用程序组件通常应该在您的应用程序容器(即简单注入器)中注册,而不是在框架的注册系统(即ServiceCollection)中注册。

然而,在注册 DbContext 时,与 ASP.NET 核心配置系统有一些紧密集成。这可以使在那里进行注册和简单地从 Simple Injector 交叉连接变得更加直接。这让配置系统可以控制 DbContext 的创建和销毁。这在执行诸如 DbContext 池化之类的事情时变得特别有价值,因为此池化功能与此配置和注册紧密耦合 API.

通常,这两个选项都非常简单,但由于您的 DbContext 实现了多个您希望单独注册的接口,这会导致您的注册变得更加复杂。如果您也使用内置的 DI 容器,情况也会如此,所以这不是 Simple Injector 特有的。

在你的例子中,纯粹在 Simple Injector 中进行注册看起来像这样:

var reg = Lifestyle.Scoped.CreateRegistration(() =>
    {
        var optionsBuilder =
            new DbContextOptionsBuilder<EntityDbContext>().UseInMemoryDatabase("Snoogans");
        return new EntityDbContext(optionsBuilder.Options);
    },
    container);

container.AddRegistration<IUnitOfWork>(reg);
container.AddRegistration<IReadEntities>(reg);
container.AddRegistration<IWriteEntities>(reg);

如果您选择从 .NET Core 配置系统交叉连接 DbContext,您的配置将如下所示:

// Add to the built-in ServiceCollection
services.AddDbContext<EntityDbContext>(options => options.UseInMemoryDatabase("Snoogans"));

// Cross-wire in Simple Injector
container.CrossWire<EntityDbContext>(app);

// Pull that registration out of Simple Injector and use it for the interface registrations
var reg = container.GetRegistration(typeof(EntityDbContext)).Registration;

// Same as before
container.AddRegistration<IUnitOfWork>(reg);
container.AddRegistration<IReadEntities>(reg);
container.AddRegistration<IWriteEntities>(reg);

如果您不使用 Simple Injector,而是纯粹使用 .NET Core 的内置 DI 容器,注册将如下所示:

services.AddDbContext<EntityDbContext>(options => options.UseInMemoryDatabase("Snoogans"));

services.AddScoped<IUnitOfWork>(c => c.GetRequiredService<EntityDbContext>());
services.AddScoped<IReadEntities>(c => c.GetRequiredService<EntityDbContext>());
services.AddScoped<IWriteEntities>(c => c.GetRequiredService<EntityDbContext>());