定义无范围的Ninject绑定以保持代码DRY

时间:2020-03-16 04:33:14

标签: c# dependency-injection ninject

我的解决方案中有一个 Web 项目和一个 Windows Service 项目。我为这2个项目创建了2个不同的绑定模块,但是您可以看到它有很多重复的代码……唯一的区别是我在{em> Web 中使用InRequestScope()项目和InTransientScope()用于 Windows服务项目。

是否可以合并绑定,并根据项目/入口点添加范围?

public class WebModule : NinjectModule
{
    public override void Load()
    {
        Bind<ApplicationDbContext>().ToSelf().InRequestScope();
        Bind<IMyRepository>().To<MyRepository>().InRequestScope();
        // more types ... InRequetScope();
    }
}

public class ServiceModule : NinjectModule
{
    public override void Load()
    {
        Bind<ApplicationDbContext>().ToSelf().InTransientScope();
        Bind<IMyRepository>().To<MyRepository>().InTransientScope();
        // more types ... InTransientScope();
    }
}

1 个答案:

答案 0 :(得分:1)

更新:

正如ninject team所述,在两种情况下我们都可以使用InRequestScope() ...由于Windows Service项目中没有Request的概念,因此ninject将使用默认范围,即{{ 1}}。


原始答案

我想出的最好的解决方案是创建一个扩展方法:

InTransientScope()

并动态设置范围。

public static class NinjectExtensions
{
    public static IBindingNamedWithOrOnSyntax<T> GetScopeByName<T>(this IBindingInSyntax<T> syntax, string scope)
    {
        if (scope.Equals("request", StringComparison.InvariantCultureIgnoreCase))
        {
            return syntax.InRequestScope();
        }
        else if (scope.Equals("thread", StringComparison.InvariantCultureIgnoreCase))
        {
            return syntax.InThreadScope();
        }
        else if (scope.Equals("singleton", StringComparison.InvariantCultureIgnoreCase))
        {
            return syntax.InSingletonScope();
        }
        
        return syntax.InTransientScope();
    }
}

注意:我不确定是否有更好的解决方案...这只是我想到的最干净的方法。