7

ドキュメントには、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

私はこの記事で概説されているのと同じ目標を達成しようとしています。

4

2 に答える 2

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>)));
于 2009-10-06T10:16:07.807 に答える
5

実行時までタイプがわからない場合は、MakeGenericTypeを使用してタイプをビルドできます。

var addressFlattener = _container.Resolve(typeof(IFlattener<>).MakeGenericType(typeof(Address)));
于 2009-11-09T07:08:41.997 に答える