服务定位器 (Service Locator)
Caliburn.Micro 预先捆绑了一个名为 IoC 的静态服务定位器。 对于那些不熟悉的人来说,服务定位器是一个实体,它可以为另一个实体提供服务实例,通常基于某种类型或键。服务定位器实际上是一种模式,与控制反转有关。许多人认为服务定位器是一种反模式,但就像所有模式一样,它也有自己的用例。
入门
IOC 是用于 Caliburn.Micro 中的服务定位的静态实体, 这使得 IOC 可以轻松地处理静态实体(如依赖项属性)。
public static class IoC {
public static Func<Type, string, object> GetInstance;
public static Func<Type, IEnumerable<object>> GetAllInstances;
public static Action<object> BuildUp;
public static T Get<T>(string key = null);
public static IEnumerable<T> GetAll<T>();
}
如你所见,IoC 的许多功能取决于提供它的消费者。在大多数情况下,所需的相关方法直接映射到所有依赖注入容器提供的方法 (尽管名称和功能可能不同)。
为 IoC 注入功能
Caliburn.Micro 要求 IoC 正常工作,因为它在框架级别利用它。 作为非可选服务,我们直接在 Bootstrapper
上提供可扩展点,以便将功能注入 IoC。该示例使用 Caliburn.Micro 自己的 SimpleContainer
为 IoC 注入功能。
public class CustomBootstrapper : Bootstrapper {
private SimpleContainer _container = new SimpleContainer();
//...
protected override object GetInstance(Type service, string key) {
return _container.GetInstance(service, key);
}
protected override IEnumerable<object> GetAllInstances(Type service) {
return _container.GetAllInstances(service);
}
protected override void BuildUp(object instance) {
_container.BuildUp(instance);
}
//...
}
通过将你选择的依赖容器映射到 IoC,Calburn.Micro 可以利用通过服务定位对容器进行的任何服务绑定。
在你的应用中使用 IOC
正如在入门时所述,除了一些特定领域外,许多人认为服务定位是一个反模式。在大多数情况下,你需要使用依赖注入容器。服务定位器解决的许多问题可以通过规划应用程序组合来解决。请参阅屏幕、导体和组合有关组成的更多信息。
但是,如果你仍然需要服务定位,那么 IoC 将使其变得容易。下面的代码展示了如何使用服务定位器检索或向实例注入服务。
获取单一服务
IOC 支持按类型或类型和密钥检索单个类型。所有依赖注入容器都不支持基于密钥的检索,因此密钥参数是可选的。
var windowManager = IoC.Get<IWindowManager>();
var windowManager = IoC.Get<IWindowManager>("windowManager");
获得一系列服务
IOC 也支持请求一系列服务。返回类型是 IEnumerable<T>
,其中 T 是请求的服务类型。LINQ 可用于过滤最终集合,但请注意,此时集合中的任何实体都已经实例化。
var viewModelCollection = IoC.GetAll<IViewModel>();
var viewModel = IoC.GetAll<IViewModel>().FirstOrDefault(vm => vm.GetType() == typeof(ShellViewModel));
注入实例
IOC 支持将服务注入到给定实例中。如何完成这项工作的机制留给了注入Action 的实现 BuildUp
字段。框架中有许多地方用于将功能注入到映射到 IOC 的依赖容器外部创建的实体中。
var viewModel = new LocallyCreatedViewModel();
IoC.BuildUp(viewModel);