具有Open Generics的Autofac和在运行时指定的类型

Pag*_*age 7 dependency-injection ioc-container autofac

该文档指出Autofac支持开放式泛型,我能够在如下基本情况下注册和解决:

注册:

builder.RegisterGeneric(typeof(PassThroughFlattener<>))
       .As(typeof(IFlattener<>))
       .ContainerScoped();
Run Code Online (Sandbox Code Playgroud)

解决:

var flattener = _container.Resolve<IFlattener<Address>>();
Run Code Online (Sandbox Code Playgroud)

上面的代码工作得很好.但是,假设我不知道在运行之前提供给IFlattener的类型,我想做这样的事情:

object input = new Address();
var flattener = (IFlattener)_container.Resolve(typeof(IFlattener<>), new TypedParameter(typeof(IFlattener<>), input.GetType()));
Run Code Online (Sandbox Code Playgroud)

这可以用AutoFac吗?我从使用StructureMap获得了以下想法:

http://structuremap.sourceforge.net/Generics.htm

我正在努力实现本文中概述的相同目标.

Pet*_*old 10

Autofac当然可以实现这一点.在"注册时间",这是你基本上做的:

  1. 注册开放泛型类型(PassThroughFlattener <>)
  2. 注册任何特定类型(AddressFlattener)
  3. 注册一个可用于根据输入对象解析IFlattener的方法

在"解决时间",您将:

  1. 解决方法
  2. 使用输入参数调用方法以解析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 closedType =
            openType.MakeGenericType(theObject.GetType());
            return (IFlattener) context.Resolve(closedType,
                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>)));
Run Code Online (Sandbox Code Playgroud)


and*_*nov 5

如果您直到运行时才知道类型,您可以使用 MakeGenericType 构建它:

var addressFlattener = _container.Resolve(typeof(IFlattener<>).MakeGenericType(typeof(Address)));
Run Code Online (Sandbox Code Playgroud)