@H_
404_0@
该文档指出Autofac
支持开放式泛型,我能够在如下基本情况下
注册和
解决:
注册:
builder.RegisterGeneric(typeof(PassThroughFlattener<>))
.As(typeof(IFlattener<>))
.ContainerScoped();
解决:
var flattener = _container.Resolve<IFlattener<Address>>();
上面的代码工作得很好.但是,假设我不知道在运行之前提供给IFlattener的类型,我想做这样的事情:
object input = new Address();
var flattener = (IFlattener)_container.Resolve(typeof(IFlattener<>),new TypedParameter(typeof(IFlattener<>),input.GetType()));
这可以用AutoFac吗?我从使用StructureMap获得了以下想法:
http://structuremap.sourceforge.net/Generics.htm
我正在努力实现本文中概述的相同目标.
Autofac当然可以实现这一点.在“
注册时间”,这是你基本上做的:
>注册开放通用类型(PassThroughFlattener<>)
>注册任何特定类型(AddressFlattener)
>注册一个可用于根据输入对象解析IFlattener的方法
在“解决时间”,您将:
>解决方法
>使用输入参数调用方法以解析IFlattener实现
这是一个(希望)工作样本:
var openType = typeof(IFlattener<>);
var builder = new ContainerBuilder();
builder.RegisterGeneric(typeof(PassThroughFlattener<>)).As(openType);
builder.Register<AddressFlattener>().As<IFlattener<Address>>();
builder.Register<Func<object,IFlattener>>(context => theObject =>
{
var concreteType =
openType.MakeGenericType(theObject.GetType());
return (IFlattener) context.Resolve(concreteType,new PositionalParameter(0,theObject));
});
var c = builder.Build();
var factory = c.Resolve<Func<object,IFlattener>>();
var address = new Address();
var addressService = factory(address);
Assert.That(addressService,Is.InstanceOfType(typeof(AddressFlattener)));
var anything = "any other data";
var anyService = factory(anything);
Assert.That(anyService,Is.InstanceOfType(typeof(PassThroughFlattener<string>)));