使用InRequestScope的Ninject与Hangfire

4
我已经将Hangfire.Ninject包安装到ASP MVC 5应用程序中,以便可以运行一些后台作业。
我已经阅读了文档,但不知道如何实现它。
我的现有配置使用InRequestScope为我的IUnitOfwork类设置,以确保每个HTTP请求只实例化一个实例,如下所示:
private static void RegisterServices(IKernel kernel)
{
        kernel.Bind<IUnitOfWork>().To<UnitOfWork>().InRequestScope();
 } 

为了在遵循文档后使用ninject与hangfire,我已经更新了我的ninjectwebcommon.cs类中的配置如下:
 private static IKernel CreateKernel()
 {
        var kernel = new StandardKernel();
        try
        {
            kernel.Bind<Func<IKernel>>().ToMethod(ctx => () => new Bootstrapper().Kernel);
            kernel.Bind<IHttpModule>().To<HttpApplicationInitializationHttpModule>();

            GlobalConfiguration.Configuration.UseNinjectActivator(kernel);

            RegisterServices(kernel);

        return kernel;
        }
        catch
        {
            kernel.Dispose();
            throw;
        }
    }

    private static void RegisterServices(IKernel kernel)
    {
         kernel.Bind<IUnitOfWork>()
            .ToSelf()
            .InNamedOrBackgroundJobScope(context => context.Kernel.Components.GetAll<INinjectHttpApplicationPlugin>()
            .Select(c => c.GetRequestScope(context))
            .FirstOrDefault(s => s != null));

    }

但现在我遇到了以下错误:
Error activating IUnitOfWork using self-binding of IUnitOfWork
No constructor was available to create an instance of the implementation type.

我有一个类想要使用hangfire来处理后台任务,代码如下:

 public class EmailJob
{
    private readonly IUnitOfWork _unitOfWork;
    private readonly IMailer _mailer;

    public EmailJob(IUnitOfWork unitOfWork, IMailer mailer)
    {
        _unitOfWork = unitOfWork;
        _notificationMailer = notificationMailer;
    }

     public void Execute()
    {
        // DO Stuff
    }

}

有人知道我做错了什么吗?文档也说明了:

任何人都可以参考文档,

Services registered with InRequestScope() directive will be unavailable during job activation, you should re-register these services without this hint.

这是什么意思?我仍然希望确保每个HTTP请求只使用一个实现dbContext的IUnitOfWork类。如果我去掉InRequestScope,这将如何影响应用程序的其余部分?
1个回答

1
我认为问题在于您将IUnitOfWork绑定到自身。类似UnitOfWork的东西需要一个具体的类来激活Niject。
  kernel.Bind<IUnitOfWork>()
        .To<UnitOfWork()
        .InNamedOrBackgroundJobScope(context => context.Kernel.Components.GetAll<INinjectHttpApplicationPlugin>()
        .Select(c => c.GetRequestScope(context))
        .FirstOrDefault(s => s != null));

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接