我希望拦截实现某个接口或具有特定属性的实例的创建。我能够对拦截扩展做类似的事情,但这似乎只做方法和属性拦截。
以下是我如何拦截方法和属性调用,但它不会拦截构造函数调用:
_kernel.Bind<IInterceptor>().To<LogInterceptor>().InSingletonScope();
_kernel.Intercept(x =>
{
if (x.Plan.Type.GetInterface(typeof(ITriggerLoggingInterception).FullName) != null)
{
return true;
}
return false;
}).With<LogInterceptor>();
正如您自己发现的那样,最接近于对每个绑定执行实例化操作(无需更改绑定)的是 IActivationStrategy。
例如(示例取自此处:
public class StartableStrategy : ActivationStrategy
{
public override void Activate(IContext context, InstanceReference reference)
{
reference.IfInstanceIs<IStartable>(x => x.Start());
}
public override void Deactivate(IContext context, InstanceReference reference)
{
reference.IfInstanceIs<IStartable>(x => x.Stop());
}
}
它通过以下方式添加到 ninject 内核中:
kernel.Components.Add<IActivationStrategy, StartableActivationStrategy>();
替代 - 结合语法糖
让我更详细地介绍我在评论中提到的OnActivation()
扩展:
public static IBindingOnSyntax<T> RegisterEvents<T>(this IBindingOnSyntax<T> binding)
{
// todo check whether <T> implements the IHandle<> interface, if not throw exception
return binding
.OnActivation((ctx, instance) => ctx.Kernel.Get<EventAggregator>().Subscribe(instance));
}
这将手动用于绑定:
kernel.Bind<FooViewModel>().ToSelf()
.RegisterEvents()
.InSingletonScope();
(不需要InSingletonScope()
- 它只是为了表明您可以像以前一样使用其他绑定扩展/功能)。
现在我认为你想"按照惯例"使用它。如果按约定 (ninject.extensions.Convention) 创建绑定,则可以使用IBindingGenerator
相应地创建绑定(无论是否调用 RegisterEvents
)。如果没有,那就更棘手了。我会说你必须扩展ninject的管道。