如何在后台任务中使用具有依赖性的作用域服务
How to consume scoped service with dependency in a background task
这是我的场景。我想使用后台任务向订阅用户发送时事通讯。这是由 MailService 完成的,它具有 UnitOfWork 作为依赖项。
我尝试了 docs.microsoft.com 的解决方案
所以在我的例子中,我使用了 IMailService 的方法而不是 ILogger,但是我得到了一个错误:
System.InvalidOperationException: 'Cannot consume scoped service >'Fit4You.Core.Data.IUnitOfWork' from singleton >'Microsoft.AspNetCore.Hosting.Internal.HostedServiceExecutor'.'
我不想让我的 UnitOfWork 或 DbContext 具有单例生命周期。 是否有可能以某种方式消耗 MailService 的依赖项,它是作用域 UnitOfWork?
我知道 IServiceScopeFactory 但可能不知道如何正确使用它。
我正在使用 .NET Core 2.2 并在接口 IHostedService 中构建
ScopedMailService:
public class ScopedMailService : IScopedMailService
{
private readonly IMailService mailService;
public ScopedMailService(IMailService mailService)
{
this.mailService = mailService;
}
public void DoWork()
{
mailService.SendNewsletterToSubscribedUsers();
}
}
ConsumeScopedMailService:
public class ConsumeScopedMailService : IHostedService
{
private Timer timer;
private readonly IMailService mailService;
public IServiceProvider Services { get; }
public ConsumeScopedMailService(IServiceProvider services, IMailService mailService)
{
Services = services;
this.mailService = mailService;
}
public Task StartAsync(CancellationToken cancellationToken)
{
var startTimeSpan = GetStartTimeSpan();
var periodTimeSpan = TimeSpan.FromSeconds(30);
timer = new Timer(DoWork, null, startTimeSpan, periodTimeSpan);
return Task.CompletedTask;
}
private void DoWork(object state)
{
using (var scope = Services.CreateScope())
{
var scopedMailService = scope.ServiceProvider.GetRequiredService<IScopedMailService>();
scopedMailService.DoWork();
}
}
public Task StopAsync(CancellationToken cancellationToken)
{
timer?.Change(Timeout.Infinite, 0);
return Task.CompletedTask;
}
public void Dispose()
{
timer?.Dispose();
}
private TimeSpan GetStartTimeSpan()
{
var currentTime = DateTime.Now.Ticks;
var executeTime = DateTime.Today.AddHours(8)
.AddMinutes(0)
.Ticks;
long ticks = executeTime - currentTime;
if (ticks < 0)
{
ticks = ticks + TimeSpan.TicksPerDay;
}
var startTimeSpan = new TimeSpan(ticks);
return startTimeSpan;
}
}
Startup.cs:
public void ConfigureServices(IServiceCollection services)
{
...
services.AddDbContext<Fit4YouDbContext>(options =>
options.UseSqlServer(Configuration.GetConnectionString("MyConnectionString")));
services.AddScoped<IUnitOfWork, UnitOfWork>();
services.AddTransient<IMailService, MailService>();
services.AddHostedService<ConsumeScopedMailService>();
services.AddScoped<IScopedMailService, ScopedMailService>();
...
}
邮件服务:
public class MailService : IMailService
{
private readonly IUnitOfWork unitOfWork;
public MailService(IUnitOfWork unitOfWork)
{
this.unitOfWork = unitOfWork;
}
public void SendNewsletterToSubscribedUsers()
{
// Some Code
}
}
单例 ConsumeScopedMailService
通过其构造函数依赖于 IMailService
:
public ConsumeScopedMailService(IServiceProvider services, IMailService mailService)
IMailService
可能是暂时的,但实现它的 class 依赖于 scoped 服务 IUnitOfWork
。间接地,ConsumeScopedMailService
最终取决于范围内的服务。
要解决此问题,应删除 IMailService mailService
。无论如何,它没有在发布的代码中使用。
这是我的场景。我想使用后台任务向订阅用户发送时事通讯。这是由 MailService 完成的,它具有 UnitOfWork 作为依赖项。
我尝试了 docs.microsoft.com 的解决方案 所以在我的例子中,我使用了 IMailService 的方法而不是 ILogger,但是我得到了一个错误:
System.InvalidOperationException: 'Cannot consume scoped service >'Fit4You.Core.Data.IUnitOfWork' from singleton >'Microsoft.AspNetCore.Hosting.Internal.HostedServiceExecutor'.'
我不想让我的 UnitOfWork 或 DbContext 具有单例生命周期。 是否有可能以某种方式消耗 MailService 的依赖项,它是作用域 UnitOfWork?
我知道 IServiceScopeFactory 但可能不知道如何正确使用它。
我正在使用 .NET Core 2.2 并在接口 IHostedService 中构建
ScopedMailService:
public class ScopedMailService : IScopedMailService
{
private readonly IMailService mailService;
public ScopedMailService(IMailService mailService)
{
this.mailService = mailService;
}
public void DoWork()
{
mailService.SendNewsletterToSubscribedUsers();
}
}
ConsumeScopedMailService:
public class ConsumeScopedMailService : IHostedService
{
private Timer timer;
private readonly IMailService mailService;
public IServiceProvider Services { get; }
public ConsumeScopedMailService(IServiceProvider services, IMailService mailService)
{
Services = services;
this.mailService = mailService;
}
public Task StartAsync(CancellationToken cancellationToken)
{
var startTimeSpan = GetStartTimeSpan();
var periodTimeSpan = TimeSpan.FromSeconds(30);
timer = new Timer(DoWork, null, startTimeSpan, periodTimeSpan);
return Task.CompletedTask;
}
private void DoWork(object state)
{
using (var scope = Services.CreateScope())
{
var scopedMailService = scope.ServiceProvider.GetRequiredService<IScopedMailService>();
scopedMailService.DoWork();
}
}
public Task StopAsync(CancellationToken cancellationToken)
{
timer?.Change(Timeout.Infinite, 0);
return Task.CompletedTask;
}
public void Dispose()
{
timer?.Dispose();
}
private TimeSpan GetStartTimeSpan()
{
var currentTime = DateTime.Now.Ticks;
var executeTime = DateTime.Today.AddHours(8)
.AddMinutes(0)
.Ticks;
long ticks = executeTime - currentTime;
if (ticks < 0)
{
ticks = ticks + TimeSpan.TicksPerDay;
}
var startTimeSpan = new TimeSpan(ticks);
return startTimeSpan;
}
}
Startup.cs:
public void ConfigureServices(IServiceCollection services)
{
...
services.AddDbContext<Fit4YouDbContext>(options =>
options.UseSqlServer(Configuration.GetConnectionString("MyConnectionString")));
services.AddScoped<IUnitOfWork, UnitOfWork>();
services.AddTransient<IMailService, MailService>();
services.AddHostedService<ConsumeScopedMailService>();
services.AddScoped<IScopedMailService, ScopedMailService>();
...
}
邮件服务:
public class MailService : IMailService
{
private readonly IUnitOfWork unitOfWork;
public MailService(IUnitOfWork unitOfWork)
{
this.unitOfWork = unitOfWork;
}
public void SendNewsletterToSubscribedUsers()
{
// Some Code
}
}
单例 ConsumeScopedMailService
通过其构造函数依赖于 IMailService
:
public ConsumeScopedMailService(IServiceProvider services, IMailService mailService)
IMailService
可能是暂时的,但实现它的 class 依赖于 scoped 服务 IUnitOfWork
。间接地,ConsumeScopedMailService
最终取决于范围内的服务。
要解决此问题,应删除 IMailService mailService
。无论如何,它没有在发布的代码中使用。