同一个服务的不同实现可以以类型,名称和键区分。
autofac提供三种典型的方式区分服务。
类型是描述服务的基本方法
builder.Register<OnlineState>().As<IDeviceState>();
这个例子将IDeviceState 类型的服务和OnlineState 的component联系起来。可以用这个服务的类型来检索OnlineState的实例:
var r = container.Resolve<IDeviceState>();
指定类型的服务在自动装配中也会起作用。
服务可以进一步按名字识别。使用这种技术时,用 Named()注册方法代替As():
builder.Register<OnlineState>().Named<IDeviceState>("online");
使用 ResolveNamed()检索指定名字的服务:
var r = container.ResolveNamed<IDeviceState>("online");
在之前的2.3版本中,ResolveNamed()只是Resolve()的简单重载。
指定名字的服务其实是指定键的服务的简单版本,所以在下一节讲到的技术也同样适用于指定名字的服务。
在一些情况下使用字符串做component的名字很方便,但是在另外一些情况下我们需要使用其他类型做键。按键指定服务提供了这种机制。
比如,使用一个枚举来描述设备的不同状态
public enum DeviceState { Online, Offline }
每个枚举值对应一个服务的实现。
public class OnlineState : IDeviceState { }
可以像下面这样使用枚举值来做键
var builder = new ContainerBuilder(); builder.RegisterType<OnlineState>().Keyed<IDeviceState>(DeviceState.Online); builder.RegisterType<OfflineState>().Keyed<IDeviceState>(DeviceState.Offline); // Register other components here
显式检索
可以使用ResolveKeye()进行显式检索。
var r = container.ResolveKeyed<IDeviceState>(DeviceState.Online);
这么做导致容器被当做 Service Locator使用,这是不被推荐的。应该使用IIndex type替代。
在之前的2.3版本中, ResolveKeyed()只是Resolve()的简单重载。
使用索引检索
Autofac.Features.Indexed.IIndex<K,V>是autofac自动实现的一个关联类型。component可以使用IIndex<K,V>作为参数的构造函数从基于键的服务中选择需要的实现。
public class Modem : IHardwareDevice { IIndex<DeviceState, IDeviceState> _states; IDeviceState _currentState; public Modem(IIndex<DeviceState, IDeviceState> states) { _states = states; SwitchOn(); } void SwitchOn() { _currentState = _states[DeviceState.Online]; } }
在SwitchOn方法中。使用索引从前面用DeviceState.Online做键注册的IDeviceState的实现。