我一直在使用Ninject,发现与我合作很愉快。一切都在代码中设置,语法相当简单,并且它具有良好的文档(以及关于SO的大量答案)。
所以基本上它是这样的:
创建视图模型,并将IStorage
接口作为构造函数参数:
class UserControlViewModel
{
public UserControlViewModel(IStorage storage)
{
}
}
ViewModelLocator
为视图模型创建一个带有get属性的对象,该属性会从Ninject加载视图模型:
class ViewModelLocator
{
public UserControlViewModel UserControlViewModel
{
get { return IocKernel.Get<UserControlViewModel>();} // Loading UserControlViewModel will automatically load the binding for IStorage
}
}
使ViewModelLocator
在App.xaml中应用广泛的资源:
<Application ...>
<Application.Resources>
<local:ViewModelLocator x:Key="ViewModelLocator"/>
</Application.Resources>
</Application>
绑定DataContext
的UserControl
在ViewModelLocator相应的属性。
<UserControl ...
DataContext="{Binding UserControlViewModel, Source={StaticResource ViewModelLocator}}">
<Grid>
</Grid>
</UserControl>
创建一个继承NinjectModule的类,该类将设置必要的绑定(IStorage
和viewmodel):
class IocConfiguration : NinjectModule
{
public override void Load()
{
Bind<IStorage>().To<Storage>().InSingletonScope(); // Reuse same storage every time
Bind<UserControlViewModel>().ToSelf().InTransientScope(); // Create new instance every time
}
}
在应用程序启动时使用必要的Ninject模块(目前上面的模块)初始化IoC内核:
public partial class App : Application
{
protected override void OnStartup(StartupEventArgs e)
{
IocKernel.Initialize(new IocConfiguration());
base.OnStartup(e);
}
}
我使用了一个静态IocKernel
类来保存IoC内核的应用程序范围的实例,因此我可以在需要时轻松访问它:
public static class IocKernel
{
private static StandardKernel _kernel;
public static T Get<T>()
{
return _kernel.Get<T>();
}
public static void Initialize(params INinjectModule[] modules)
{
if (_kernel == null)
{
_kernel = new StandardKernel(modules);
}
}
}
此解决方案确实使用了静态ServiceLocator
(IocKernel
),通常将其视为反模式,因为它隐藏了类的依赖关系。但是,避免对UI类进行某种形式的手动服务查找非常困难,因为它们必须具有无参数的构造函数,而且您无论如何也无法控制实例化,因此无法注入VM。至少通过这种方式,您可以隔离地测试VM,这是所有业务逻辑所在的位置。
如果有人有更好的方法,请分享。
编辑:Lucky Likey通过让Ninject实例化UI类提供了摆脱静态服务定位器的答案。答案的细节可以在这里看到