所以我使用了一个带有属性和过滤器的简单存储库模式,因为我使用的是来自NuGet的Ninject.Web.WebApi-RC
软件包。{/ 3}
这适用于第一个请求,但由于我在请求范围内有DbContext
,因此会在所有后续请求中处理。
这是我的属性:
public class CommunicationKeyValidationAttribute : FilterAttribute
{
}
这是我的过滤器:
public class CommunicationKeyValidationFilter : AbstractActionFilter
{
public CommunicationKeyValidationFilter(IRepository repository)
{
this.repository = repository;
}
public override void OnActionExecuting(HttpActionContext actionContext)
{
// do stuff
}
}
这是我的存储库:
public class Repository : IRepository
{
public Repository(MyDbContext dbContext)
{
this.dbContext = dbContext;
}
}
以下是我的Ninject绑定:
this.Kernel.Bind<MyDbContext>().ToSelf().InRequestScope();
this.Kernel.Bind<IRepository>().To<Repository>().InRequestScope();
this.Kernel.BindHttpFilter<CommunicationKeyValidationFilter>(FilterScope.Action)
.WhenActionMethodHas<CommunicationKeyValidationAttribute>()
.InRequestScope();
我的控制器看起来像这样:
public class HomeController
{
[CommunicationKeyValidation]
public ActionResult Index()
{
// do stuff
}
这里的问题是CommunicationKeyValidationFilter
上的构造函数仅在第一个请求时被调用。有没有办法可以在每次尝试解析过滤器时让ninject构造这个过滤器?
答案 0 :(得分:15)
过滤器由WebApi缓存。它们应该处于瞬态范围内,以便WebApi可以管理生命周期。由于生命周期较长,因此您不能拥有任何生命周期较短的依赖项。
您可以做的是在执行过滤器期间创建存储库。为此,最好使用NinjectFactoryExtension
注入工厂:
public class CommunicationKeyValidationFilter : AbstractActionFilter
{
public CommunicationKeyValidationFilter(IRepositoryFactory repositoryFactory)
{
this.repositoryFactory = repositoryFactory;
}
public override void OnActionExecuting(HttpActionContext actionContext)
{
var repository = this.repositoryFactory.CreateRepository();
}
}
public interface IRepositoryFactory { IRepository CreateRepository(); }
kernel.Bind<IRepositoryFactory>().ToFactory();