- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我有一个 .NET MVC 5 .NET Framework 应用程序,我正在将其转换为 .NET Core 2.1
我有一个自定义操作过滤器,它在 .NET Framework 版本中被注册为 Filterconfig 类中的全局过滤器,如下所示:
public class FilterConfig
{
public static void RegisterGlobalFilters(GlobalFilterCollection filters)
{
filters.Add(new MyCustomActionFilter());
}
}
在 .NET 版本的自定义操作过滤器中,我使用的是服务定位器模式(我知道它可以被认为是一种反模式),如下所示:
var myService = DependencyResolver.Current.GetService<IMyService>();
我正在为 DI 使用 Simple Injector,在 .NET 版本中一切正常。在 .NET Core 版本中,我试图获得相同的功能,但 myService 始终为 null
我仍在使用 Simple Injector(因为解决方案中的所有其他项目都在使用它,并且它们没有转移到 .NET Core 项目(只有 Web 项目是)。
我的 Startup.cs 类有这段代码:
services.Configure<MvcOptions>(options =>
{
options.Filters.Add(new MyCustomActionFilter());
});
SimpleInjectorConfig.IntegrateSimpleInjector(services, container);
在我的服务层,我有一个从 Web 层调用的 SimpleInjector Registartion 类 - 然后它向下调用 DAL 层进行注册
public class SimpleInjectorRegistration
{
public static void RegisterServices(Container container)
{
container.Register<IMyService, MyService>();
//further code removed for brevity
当我使用自定义过滤器中的断点和此 RegisterServices 方法中的断点运行应用程序时,我可以看到首先命中 RegisterServices 方法中的断点,然后是自定义过滤器中的断点 - 这让我觉得一切都已连线正确放入容器中。
但是,我正在尝试使用 .NET Core 服务定位器模式在自定义过滤器中再次执行以下操作
var myService = filterContext.HttpContext.RequestServices.GetService<IMyService>();
但是结果总是空的?
我在这个设置中遗漏了什么吗?
------------ 更新--------------------
根据史蒂文斯的评论,我向我的操作过滤器添加了一个构造函数,并传入了 Simple Injector 容器。
现在我的 Startup 类是:
public class Startup
{
//Simple Injector container
private Container container = new Container();
public Startup(IConfiguration configuration)
{
Configuration = configuration;
}
public IConfiguration Configuration { get; }
public void ConfigureServices(IServiceCollection services)
{
JwtSecurityTokenHandler.DefaultInboundClaimTypeMap.Clear();
services.Configure<MvcOptions>(options =>
{
options.Filters.Add(new MyCustomActionFilter(container));
我的自定义过滤器现在如下所示,添加了构造函数:
public class MyCustomActionFilter : ActionFilterAttribute
{
private readonly IMyService _myService;
public MyCustomActionFilter(Container container)
{
_myService = container.GetService<IMyService>();
}
public override void OnActionExecuting(ActionExecutingContext filterContext)
{
//actual code of custom filter removed - use of MyService
我在 MyCustomActionFilter 的构造函数上设置了一个断点,我可以看到它被击中,但我抛出了一个错误:
SimpleInjector.ActivationException: 'The IDbContext is registered as 'Async Scoped' lifestyle, but the instance is requested outside the context of an active (Async Scoped) scope.'
MyService 对注入(inject)其中的 DbContext 有依赖性(它正在执行从数据库保存和检索数据的工作。
对于数据库上下文,我注册如下:
public class SimpleInjectorRegistration
{
public static void RegisterServices(Container container, string connectionString)
{
container.Register<IDbContext>(() => new MyDbContext(connectionString),
Lifestyle.Scoped);
}
}
最佳答案
在旧的 ASP.NET MVC 和新的 ASP.NET Core 中如何集成 Simple Injector 之间存在一些重大变化。在旧系统中,您可以替换 IDependencyResolver
。然而,ASP.NET Core 包含一个完全不同的模型,它有自己的内部 DI 容器。原样impossible要用 Simple Injector 替换内置容器,您将让两个容器并排运行。在这种情况下,内置容器将解析框架和第三方组件,而 Simple Injector 将为您组合应用程序组件。
当您调用 HttpContext.RequestServices.GetService
时,您将请求内置容器的服务,不是简单注入(inject)器。将 IMyService
注册添加到内置容器中,正如 TanvirArjel 的回答所暗示的那样,一开始似乎可行,但这完全从等式中跳过了 Simple Injector,这显然不是您希望的一种选择使用 Simple Injector 作为您的应用程序容器。
要模仿您之前的类似服务定位器的行为,您必须将 SimpleInjector.Container
注入(inject)到您的过滤器中,如下所示:
options.Filters.Add(new MyCustomActionFilter(container));
但是,如您在问题中所示,从构造函数中调用容器是错误的:
public class MyCustomActionFilter : ActionFilterAttribute
{
private readonly IMyService _myService;
public MyCustomActionFilter(Container container)
{
_myService = container.GetService<IMyService>(); // NEVER DO THIS!!!
}
...
}
WARNING: You should never resolve from the container from the constructor. Or in more general: you should never use any injected dependency from inside the constructor. The constructor should only store the dependency.
正如 Mark Seemann 所解释的那样,injection constructors should be simple .在这种情况下,情况甚至会变得更糟,因为:
MyCustomActionFilter
的构造函数期间,没有事件作用域,无法解析IMyService
IMyService
可以解析,MyCustomActionFilter
也是一个 Singleton
并且将 IMyService
存储在私有(private)字段中将导致隐藏 Captive Dependency .这可能会导致各种麻烦。您应该存储Container
依赖项,而不是存储已解析的IMyService
依赖项:
public class MyCustomActionFilter : ActionFilterAttribute
{
private readonly Container _container;
public MyCustomActionFilter(Container container)
{
_container = container;
}
public override void OnActionExecuting(ActionExecutingContext filterContext)
{
myService = container.GetService<IMyService>();
//actual code of custom filter removed - use of MyService
}
}
在调用 OnActionExecuting
期间,将有一个事件的简单注入(inject)器 Scope
,这将允许解析 IMyService
。最重要的是,由于 IMyService
未存储在私有(private)字段中,因此不会被缓存,也不会导致 Captive Dependency。
在您的问题中,您提到了 Service Locator anti-pattern .将 Container
注入(inject)过滤器是否实际上是服务定位器反模式的实现取决于过滤器所在的位置。正如马克·西曼 puts它:
A DI container encapsulated in a Composition Root is not a Service Locator - it's an infrastructure component.
换句话说,只要过滤器类位于内部您的Composition Root ,您没有应用服务定位器反模式。然而,这确实意味着您必须确保过滤器本身包含尽可能少的有趣行为。该行为应全部移至过滤器解析的服务。
关于c# - 带有返回 null 的简单注入(inject)器的 ASP.NET Core 2.1 服务定位器,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/53637611/
我正在尝试测试依赖于其他服务 authService 的服务 documentViewer angular .module('someModule') .service('docu
如果我的网站上线(不要认为它会,目前它只是一个学习练习)。 我一直在使用 mysql_real_escape_string();来自 POST、SERVER 和 GET 的数据。另外,我一直在使用 i
我有以下代码,它容易受到 SQL 注入(inject)的攻击(我认为?): $IDquery = mysqli_query($connection, "SELECT `ID` FROM users W
我一直在自学如何创建扩展,以期将它们用于 CSS 注入(inject)(以及最终以 CSS 为载体的 SVG 注入(inject),但那是以后的问题)。 这是我当前的代码: list .json {
这个简单的代码应该通过 Java Spring 实现一个简单的工厂。然而结果是空指针,因为 Human 对象没有被注入(inject)对象(所以它保持空)。 我做错了什么? 谢谢 配置 @Config
我正在编写一个 ASP.NET MVC4 应用程序,它最终会动态构建一个 SQL SELECT 语句,以便稍后存储和执行。动态 SQL 的结构由用户配置以用户友好的方式确定,具有标准复选框、下拉列表和
首先让我说我是我为确保 SQL 注入(inject)攻击失败而采取的措施的知己。所有 SQL 查询值都是通过事件记录准备语句完成的,所有运算符(如果不是硬编码)都是通过数字白名单系统完成的。这意味着如
这是 SQL 映射声称可注入(inject)的负载: user=-5305' UNION ALL SELECT NULL,CONCAT(0x716b6b7071,0x4f5577454f76734
我正在使用 Kotlin 和 Android 架构组件(ViewModel、LiveData)构建一个新的 Android 应用程序的架构,并且我还使用 Koin 作为我的依赖注入(inject)提供
假设 RequestScope 处于 Activity 状态(使用 cdi-unit 的 @InRequestScope) 给定 package at.joma.stackoverflow.cdi;
我有一个搜索表单,可以在不同的提供商中搜索。 我从拥有一个基本 Controller 开始 public SearchController : Controller { protected r
SQLite 注入 如果您的站点允许用户通过网页输入,并将输入内容插入到 SQLite 数据库中,这个时候您就面临着一个被称为 SQL 注入的安全问题。本章节将向您讲解如何防止这种情况的发生,确保脚
我可以从什么 dll 中获得 Intercept 的扩展?我从 http://github.com/danielmarbach/ninject.extensions.interception 添加了
使用 NInject 解析具有多个构造函数的类似乎不起作用。 public class Class1 : IClass { public Class1(int param) {...} public
我有一个 MetaManager 类: @Injectable() export class MetaManager{ constructor(private handlers:Handler
我是 Angular 的新手,我不太清楚依赖注入(inject)是如何工作的。我的问题是我有依赖于服务 B 的服务 A,但是当我将服务 A 注入(inject)我的测试服务 B 时,服务 B 变得未定
我正在为我的项目使用 android 应用程序启动、刀柄和空间。我在尝试排队工作时遇到错误: com.test E/WM-WorkerFactory: Could not instantiate co
我不确定这是什么糖语法,但让我向您展示问题所在。 def factors num (1..num).select {|n| num % n == 0} end def mutual_factors
简单的问题,我已经看过这个了:Managing imports in Scalaz7 ,但我不知道如何最小化注入(inject) right和 left方法到我的对象中以构造 \/ 的实例. 我确实尝
在我的 Aurelia SPA 中,我有一些我想在不同模块中使用的功能。它依赖于调用时给出的参数和单例的参数。有没有办法创建一个导出函数,我可以将我的 Auth 单例注入(inject)其中,而不必在
我是一名优秀的程序员,十分优秀!