NamedScopes Ninject绑定和异步(线程)

时间:2012-07-29 16:57:31

标签: c# asp.net-mvc ninject

我的项目由服务和存储库构成(所有存储库共享数据库上下文)。在我的一个服务层中,我有一个使用存储库写入数据库的异步方法。在此方法可以使用之前,Web请求将完成并处理上下文。我尝试了解NamedScopes中所述的answer。我似乎还无法理解如何实现它。我将展示我的项目是如何构建的,并希望有人可以在代码级别帮助我。

绑定

    private static void RegisterServices(IKernel kernel)
    {
        //dbcontext
        kernel.Bind<EntityDatabaseContext>().ToMethod(context => new EntityDatabaseContext()).InRequestScope();

        //unit of work
        kernel.Bind<IUnitOfWork>().To<UnitOfWork>().InRequestScope();

        //repositories
        kernel.Bind<IRepository<Account>>().To<Repository<Account>>().InRequestScope();

        //services
        kernel.Bind<IAuthenticationService>().To<AuthenticationService>().InRequestScope();
    }

AuthenticationService使用构造函数注入

public AuthenticationService(UnitOfWork unitOfWork, IRepository<Account> accountRepository){}

我的AuthenticationService中的方法

    //this is a background process
    public Task SomeMethodAsync(string text)
    {
        //spin it off into a new task
        return Task.Factory.StartNew(() => SomeMethod(text));
    }

SomeMethod使用accountRepository。如果需要更多信息,请告诉我。请帮我解决线程问题,如果NamedScopes是解决方案,我该如何在我的情况下实现它?

基本上,正在执行后台进程,它正在使用由于请求范围而由ninject处理的上下文。

1 个答案:

答案 0 :(得分:6)

您应该知道运行后台线程可能会导致很多问题。 IIS可以随时决定回收应用程序池,这将立即终止您的线程(或者在某些情况下根本不执行),使您的应用程序处于不一致状态。

http://haacked.com/archive/2011/10/16/the-dangers-of-implementing-recurring-background-tasks-in-asp-net.aspx

运行异步操作的最简单且最不容易出错的方法是实现Windows服务并将这些异步操作委托给Windows服务,例如:使用MSMQ。

如果你仍然想要努力,那么请阅读HostingEnvironment.RegisterObjectIRegisteredObject以防止这些不一致的情况。

Ninject部分非常简单。只创建一些作业处理器类,例如MyJobProcessor获取所有必需的依赖项来执行任务。它应该实现INotifyWhenDisposed。最简单的方法是从DisposeNotifyingObject派生。

public class MyJobProcessor : DisposeNotifyingObject, IRegisteredObject
{
    public void Execute() { ... }
    public void Stop(bool immediate) { ... }
}

将此处理器注入控制器,让任务启动它并在完成工作后将其处理。

Task.Factory.StartNew(() => 
    { 
        try 
        { 
            processor.Execute(); 
        } 
        finally 
        { 
            processor.Dispose); 
        }
    });

指定它是其依赖项的范围。

Bind<MyJobProcessor>().ToSelf().Named("MyJobProcessor").DefinesNamedScope("MyJobProcessorScope");
Bind<IUnitOfWork>().To<UnitOfWork>().WhenAnyAnchestorNamed("MyJobProcessor").InNamedScope("MyJobProcessorScope");