我有一个界面:
public interface IService { void DoStuff(int parm1,string parm2,Guid gimmeABreakItsAnExampleK); }
我想配置Ninject(v3)绑定,以便我可以使用“dispatcher”shuffle方法调用多个IService实例,像这样:
public sealed class DispatcherService : IService { private IEnumerable<IService> _children; public DispatcherService(IEnumerable<IService> children) { this._children = children.ToList(); } public void DoStuff(int parm1,Guid gimmeABreakItsAnExampleK) { foreach(var child in this._children) { child.DoStuff(parm1,parm2,gimmeABreakItsAnExampleK); } } }
但是,我的绑定,看起来像这样,结束在运行时抛出异常指示循环依赖:
this.Bind<IService>().To<DispatcherService>(); this.Bind<IService>().To<SomeOtherService>() .WhenInjectedExactlyInto<DispatcherService>(); this.Bind<IService>().To<YetAnotherService>() .WhenInjectedExactlyInto<DispatcherService>();
这可能吗?如果是这样,我做错了什么?忍者能逃避这种循环依赖的厄运吗?
解决方法
如果您的调度程序是唯一需要将IServices列表作为参数的IService,那么这个工作(我测试过):
kernel.Bind<IService>().To<DispatcherService>().When(x => x.IsUnique); this.Bind<IService>().To<SomeOtherService>() .WhenInjectedExactlyInto<DispatcherService>(); this.Bind<IService>().To<YetAnotherService>() .WhenInjectedExactlyInto<DispatcherService>();
在这种情况下,When子句的工作原理是当您的构造函数调用单个服务实例时,IRequest的IsUnique字段设置为true.由于您的DispatcherService调用IEnumerable,因此在激活DispatcherService时该值为false.这阻止了循环依赖的发生.
真的,告诉内核不要试图将DispatcherService注入本身的任何正确的方法都可以工作(这只是一个潜在的有用的例子).
编辑:简单地简化循环依赖关系的更明确的方式似乎是这样的:
kernel.Bind<IService>().To<DispatcherService>().When( request => request.Target.Member.DeclaringType != typeof (DispatcherService));