geo*_*rtz 53 c# asp.net-core asp.net-core-2.0
当我尝试运行我的应用程序时,我收到错误
InvalidOperationException: Cannot resolve 'API.Domain.Data.Repositories.IEmailRepository' from root provider because it requires scoped service 'API.Domain.Data.EmailRouterContext'.
Run Code Online (Sandbox Code Playgroud)
奇怪的是,这个EmailRepository和界面设置完全相同,因为我可以告诉所有其他存储库,但没有为它们抛出错误.只有在我尝试使用app.UseEmailingExceptionHandling()时才会出现错误; 线.这是我的一些Startup.cs文件.
public class Startup
{
public IConfiguration Configuration { get; protected set; }
private APIEnvironment _environment { get; set; }
public Startup(IConfiguration configuration, IHostingEnvironment env)
{
Configuration = configuration;
_environment = APIEnvironment.Development;
if (env.IsProduction()) _environment = APIEnvironment.Production;
if (env.IsStaging()) _environment = APIEnvironment.Staging;
}
public void ConfigureServices(IServiceCollection services)
{
var dataConnect = new DataConnect(_environment);
services.AddDbContext<GeneralInfoContext>(opt => opt.UseSqlServer(dataConnect.GetConnectString(Database.GeneralInfo)));
services.AddDbContext<EmailRouterContext>(opt => opt.UseSqlServer(dataConnect.GetConnectString(Database.EmailRouter)));
services.AddWebEncoders();
services.AddMvc();
services.AddScoped<IGenInfoNoteRepository, GenInfoNoteRepository>();
services.AddScoped<IEventLogRepository, EventLogRepository>();
services.AddScoped<IStateRepository, StateRepository>();
services.AddScoped<IEmailRepository, EmailRepository>();
}
public void Configure(IApplicationBuilder app, IHostingEnvironment env, ILoggerFactory loggerFactory)
{
loggerFactory.AddConsole();
app.UseAuthentication();
app.UseStatusCodePages();
app.UseEmailingExceptionHandling();
app.UseMvcWithDefaultRoute();
}
}
Run Code Online (Sandbox Code Playgroud)
这是EmailRepository
public interface IEmailRepository
{
void SendEmail(Email email);
}
public class EmailRepository : IEmailRepository, IDisposable
{
private bool disposed;
private readonly EmailRouterContext edc;
public EmailRepository(EmailRouterContext emailRouterContext)
{
edc = emailRouterContext;
}
public void SendEmail(Email email)
{
edc.EmailMessages.Add(new EmailMessages
{
DateAdded = DateTime.Now,
FromAddress = email.FromAddress,
MailFormat = email.Format,
MessageBody = email.Body,
SubjectLine = email.Subject,
ToAddress = email.ToAddress
});
edc.SaveChanges();
}
public void Dispose()
{
Dispose(true);
GC.SuppressFinalize(this);
}
private void Dispose(bool disposing)
{
if (!disposed)
{
if (disposing)
edc.Dispose();
disposed = true;
}
}
}
Run Code Online (Sandbox Code Playgroud)
最后是异常处理中间件
public class ExceptionHandlingMiddleware
{
private const string ErrorEmailAddress = "errors@ourdomain.com";
private readonly IEmailRepository _emailRepository;
private readonly RequestDelegate _next;
public ExceptionHandlingMiddleware(RequestDelegate next, IEmailRepository emailRepository)
{
_next = next;
_emailRepository = emailRepository;
}
public async Task Invoke(HttpContext context)
{
try
{
await _next.Invoke(context);
}
catch (Exception ex)
{
await HandleExceptionAsync(context, ex, _emailRepository);
}
}
private static Task HandleExceptionAsync(HttpContext context, Exception exception,
IEmailRepository emailRepository)
{
var code = HttpStatusCode.InternalServerError; // 500 if unexpected
var email = new Email
{
Body = exception.Message,
FromAddress = ErrorEmailAddress,
Subject = "API Error",
ToAddress = ErrorEmailAddress
};
emailRepository.SendEmail(email);
context.Response.ContentType = "application/json";
context.Response.StatusCode = (int) code;
return context.Response.WriteAsync("An error occured.");
}
}
public static class AppErrorHandlingExtensions
{
public static IApplicationBuilder UseEmailingExceptionHandling(this IApplicationBuilder app)
{
if (app == null)
throw new ArgumentNullException(nameof(app));
return app.UseMiddleware<ExceptionHandlingMiddleware>();
}
}
Run Code Online (Sandbox Code Playgroud)
更新:我发现此链接https://github.com/aspnet/DependencyInjection/issues/578这导致我从此更改我的Program.cs文件的BuildWebHost方法
public static IWebHost BuildWebHost(string[] args)
{
return WebHost.CreateDefaultBuilder(args)
.UseStartup<Startup>()
.Build();
}
Run Code Online (Sandbox Code Playgroud)
对此
public static IWebHost BuildWebHost(string[] args)
{
return WebHost.CreateDefaultBuilder(args)
.UseStartup<Startup>()
.UseDefaultServiceProvider(options =>
options.ValidateScopes = false)
.Build();
}
Run Code Online (Sandbox Code Playgroud)
我不知道究竟发生了什么,但它现在似乎有效.
use*_*336 103
您IEmailRepository在Startup课程中注册了作为范围的服务.这意味着您不能将其作为构造函数参数注入,Middleware因为只有Singleton服务可以通过构造函数注入来解析Middleware.您应该将依赖项移动到这样的Invoke方法:
public ExceptionHandlingMiddleware(RequestDelegate next)
{
_next = next;
}
public async Task Invoke(HttpContext context, IEmailRepository emailRepository)
{
try
{
await _next.Invoke(context);
}
catch (Exception ex)
{
await HandleExceptionAsync(context, ex, emailRepository);
}
}
Run Code Online (Sandbox Code Playgroud)
Rid*_*dik 38
获取作用域依赖项实例的另一种方法是将服务提供者(IServiceProvider)注入中间件构造函数,scope在Invoke方法中创建,然后从作用域获取所需的服务:
using (var scope = _serviceProvider.CreateScope()) {
var _emailRepository = scope.ServiceProvider.GetRequiredService<IEmailRepository>);
//do your stuff....
}
Run Code Online (Sandbox Code Playgroud)
退房解决服务中的方法体在asp.net核心依赖注入的最佳实践技巧窍门的更多细节.
Joe*_*tte 21
中间件始终是单例,因此您不能在中间件的构造函数中将作用域依赖项作为构造函数依赖项.
中间件支持对Invoke方法进行方法注入,因此您只需将IEmailRepository emailRepository作为参数添加到该方法中,它就会被注入,并且可以作为作用域使用.
public async Task Invoke(HttpContext context, IEmailRepository emailRepository)
{
....
}
Run Code Online (Sandbox Code Playgroud)
Har*_*han 13
您的middleware和service必须彼此兼容才能service通过 constructor 您的注入middleware。在这里,您middleware已创建为 a,convention-based middleware这意味着它充当 a singleton service,并且您已将服务创建为scoped-service。因此,您不能将 a 注入scoped-service到 a 的构造函数中singleton-service,因为它会强制 ascoped-service充当 1 singleton。但是,以下是您的选择。
InvokeAsync方法中。middleware一个factory-based。AFactory-based middleware能够充当scoped-service. 因此,您可以scoped-service通过该中间件的构造函数注入另一个中间件。下面,我向您展示了如何创建factory-based中间件。
这仅用于演示。所以,我删除了所有其他代码。
public class Startup
{
public Startup()
{
}
public void ConfigureServices(IServiceCollection services)
{
services.AddScoped<TestMiddleware>();
services.AddScoped<TestService>();
}
public void Configure(IApplicationBuilder app)
{
app.UseMiddleware<TestMiddleware>();
}
}
Run Code Online (Sandbox Code Playgroud)
这TestMiddleware:
public class TestMiddleware : IMiddleware
{
public TestMiddleware(TestService testService)
{
}
public Task InvokeAsync(HttpContext context, RequestDelegate next)
{
return next.Invoke(context);
}
}
Run Code Online (Sandbox Code Playgroud)
这TestService:
public class TestService
{
}
Run Code Online (Sandbox Code Playgroud)
小智 11
在 .NET Core 6 中,以下设置对我有用。
using (var scope = app.Services.CreateScope())
{
var services = scope.ServiceProvider.GetRequiredService<IDbInitilizer>;
services.Invoke().Initialize();
}
Run Code Online (Sandbox Code Playgroud)
| 归档时间: |
|
| 查看次数: |
33613 次 |
| 最近记录: |