首页 > 解决方案 > OWIN + Ninject w/ UseNinjectMiddleware 不处理

问题描述

当使用任何实现 IDisposable 的注入对象在 OWIN 中将 Ninject 内核注册为中间件组件时,UseNinjectMiddleware不会在请求结束时处理掉。

我已经放入控制台打印来观察实例何时被创建和处理。我注意到每个请求都会创建实例,但直到某个随机时间之后才会被处理掉。

我创建了一个空项目来查找根本原因。我使用了一个简单的模型来跟踪它的实例编号,这样我就可以知道正在创建和处理什么。

模型:

public class Demo : IDemo, IDisposable {
    public static int InstanceCount = 1; //Tracks the current number of 
                                         //instances that have been created.
    public Demo() {
        Id = InstanceCount++; 
        Debug.WriteLine("Initialized Instance " + Id);
    }
    public int Id { get; set; }
    public void Dispose() {
        Debug.WriteLine("Disposed instance" + Id);
        Id = -1;
    }
}

public interface IDemo {
    int Id { get; set; }
}

控制器

public class HomeController : Controller {
    private readonly IDemo demo;
    public HomeController(IDemo demo) { this.demo = demo; }

    public ActionResult Index() {
        ViewBag.Id = demo.Id;
        return View();
    }
}

启动文件

public class Startup {
    public void Configuration(IAppBuilder app) {
        var kernel = CreateKernel();
        app.UseNinjectMiddleware(() => kernel);
    }

    public IKernel CreateKernel() {
        var kernel = new StandardKernel();
        kernel.Bind<IDemo>().To<Demo>().InRequestScope();
        return kernel;
    }
}

我还确保所有软件包都是最新的,因为这是以前版本的 Ninject 中的一个已知问题。

<?xml version="1.0" encoding="utf-8"?>
<packages>
  <package id="Microsoft.AspNet.Mvc" version="5.2.7" targetFramework="net462" />
  <package id="Microsoft.AspNet.Razor" version="3.2.7" targetFramework="net462" />
  <package id="Microsoft.AspNet.WebPages" version="3.2.7" targetFramework="net462" />
  <package id="Microsoft.CodeDom.Providers.DotNetCompilerPlatform" version="2.0.1" targetFramework="net462" />
  <package id="Microsoft.Owin" version="4.0.1" targetFramework="net462" />
  <package id="Microsoft.Owin.Host.SystemWeb" version="4.0.1" targetFramework="net462" />
  <package id="Microsoft.Web.Infrastructure" version="1.0.0.0" targetFramework="net462" />
  <package id="Ninject" version="3.3.4" targetFramework="net462" />
  <package id="Ninject.MVC5" version="3.3.0" targetFramework="net462" />
  <package id="Ninject.Web.Common" version="3.3.1" targetFramework="net462" />
  <package id="Ninject.Web.Common.OwinHost" version="3.3.1" targetFramework="net462" />
  <package id="Ninject.Web.Common.WebHost" version="3.3.1" targetFramework="net462" />
  <package id="Owin" version="1.0" targetFramework="net462" />
  <package id="WebActivatorEx" version="2.2.0" targetFramework="net462" />
</packages>

预期的

查看应用程序的主页时,每个请求都应打印出来:

Initialized Instance #

然后在请求结束时

Disposed instance #

#当前实例编号在哪里

实际的

每次查看页面时,只打印初始化。

Initialized Instance #

问题

我做错了什么还是这是 Ninject 和 Owin 的错误?有没有办法解决这个问题?

标签: c#dependency-injectionowinninjectdispose

解决方案


我在 GitHub 上提取了Ninject.Web.Common存储库的副本,并在 OwinHost 项目中对 OwinBootstrapper.cs 文件进行了一些更改

我在 Execute 方法中在该行之后添加了两await next(context)

var currentContext = HttpContext.Current;
this.bootstrapper.Kernel?.Components.Get<ICache>().Clear(currentContext);
public Func<IDictionary<string, object>, Task> Execute(Func<IDictionary<string, object>, Task> next)
{
    this.bootstrapper.Initialize(() =>
    {
        var kernel = this.createKernelCallback();
        lock (this.modules)
        {
            kernel.Load(this.modules);
        }
        return kernel;
    });

    return async context =>
    {
        using (var scope = new OwinRequestScope())
        {
            context[NinjectOwinRequestScope] = scope;
            await next(context);

            var currentContext = HttpContext.Current;
            this.bootstrapper.Kernel?.Components.Get<ICache>().Clear(currentContext);
        }
    };
}

然后,我将该项目构建为本地 nuget 包,并将 nuget 上的引用包替换为一个本地包。


推荐阅读