依赖注入 – 在运行时使用Open Generics和Type指定的Autofac

该文档指出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>)));

相关文章

迭代器模式(Iterator)迭代器模式(Iterator)[Cursor]意图...
高性能IO模型浅析服务器端编程经常需要构造高性能的IO模型,...
策略模式(Strategy)策略模式(Strategy)[Policy]意图:定...
访问者模式(Visitor)访问者模式(Visitor)意图:表示一个...
命令模式(Command)命令模式(Command)[Action/Transactio...
生成器模式(Builder)生成器模式(Builder)意图:将一个对...