Sho*_*hoe 36 .net c# asp.net .net-core asp.net-core
我应该如何将一个DbContext实例注入(使用标准依赖注入)IHostedService?
我目前让我的IHostedService类MainContext从DbContext构造函数中获取(派生自)实例.
当我运行应用程序时,我得到:
无法从singleton'Microsoft.Extensions.Hosting.IHostedService'中使用作用域服务"Microsoft.EntityFrameworkCore.DbContextOptions".
所以我试着DbContextOptions通过指定来制作瞬态:
services.AddDbContext<MainContext>(options => 
                options.UseSqlite("Data Source=development.db"), ServiceLifetime.Transient);
在我Startup班上
但是错误仍然是相同的,即使根据这个解决的Github问题,DbContextOptions传递的应该具有在AddDbContext调用中指定的相同生命周期.
我无法使数据库上下文成为单例,否则对它的并发调用将产生并发异常(由于数据库上下文不保证是线程安全的事实).
Mar*_*ich 74
在托管服务中使用服务的一种好方法是在需要时创建范围.这允许使用服务/ db上下文等,并使用它们设置的生存期配置.理论上,不创建范围可能会导致创建单例DbContexts和不正确的上下文重用(EF Core 2.0与DbContext池).
为此,请注入IServiceScopeFactory并使用它在需要时创建范围.然后解决此范围内所需的所有依赖项.如果您希望将逻辑移出托管服务并使用托管服务仅触发某些工作(例如,定期触发任务 - 这将定期创建范围,创建任务服务),这也允许您将自定义服务注册为范围依赖项这个范围也会得到一个db上下文注入).
public class MyHostedService : IHostedService
{
    private readonly IServiceScopeFactory scopeFactory;
    public MyHostedService(IServiceScopeFactory scopeFactory)
    {
        this.scopeFactory = scopeFactory;
    }
    public void DoWork()
    {
        using (var scope = scopeFactory.CreateScope())
        {
            var dbContext = scope.ServiceProvider.GetRequiredService<MyDbContext>();
            …
        }
    }
    …
}
DbContext对于从单例服务消费的特定情况从 .NET 5 开始,您可以IDbContextFactory<TContext>为您的DbContext并将其注入到单例服务中。
用于AddDbContextFactory<TContext>注册工厂:
public void ConfigureServices(IServiceCollection services)
{
    services.AddDbContextFactory<MyDbContext>(
        options.UseSqlite("Data Source=development.db"));
}
AddDbContext()注意:从 .NET 6 开始,您可以在使用时删除,AddDbContextFactory()因为后者还将上下文类型本身注册为作用域服务。
在单例服务中注入IDbContextFactory<TContext>并使用它来创建需要CreateDbContext()的实例:DbContext
public class MySingletonService : BackgroundService, IHostedService
{
    private readonly IDbContextFactory<MyDbContext> _contextFactory;
    public MySingletonService(IDbContextFactory<MyDbContext> contextFactory)
    {
        _contextFactory = contextFactory;
    }
    protected override async Task ExecuteAsync(CancellationToken stoppingToken)
    {
        using (MyDbContext dbContext = _contextFactory.CreateDbContext())
        {
            await dbContext.MyData.ToListAsync(stoppingToken);
        }
    }
}
另请参阅
MS 文档:使用 DbContext 工厂(例如 Blazor)
备注
不要与以下内容混淆:
-IDbContextFactory<TContext>来自实体框架(4.3.1、5.0.0、6.2.0)。
-IDbContextFactory<TContext>来自实体框架核心(1.0、1.1、2.0、2.1、2.2)。
| 归档时间: | 
 | 
| 查看次数: | 12544 次 | 
| 最近记录: |