17

我习惯于创建自己的工厂,如图所示(为了说明而简化了):

public class ElementFactory
{
    public IElement Create(IHtml dom)
    {
        switch (dom.ElementType)
        {
            case "table":
                return new TableElement(dom);
            case "div":
                return new DivElement(dom);
            case "span":
                return new SpanElement(dom);
        }
        return new PassthroughElement(dom);
    }
}

我终于开始在我当前的项目中使用 IoC 容器(AutoFac),我想知道是否有一些神奇的方法可以用 AutoFac 优雅地实现同样的事情?

4

1 回答 1

25

简短的回答:是的。

更长的答案:首先,在类 Foo 被注册为 IFoo 的实现的简单情况下,构造函数参数或类型的属性Func<IFoo>将由 Autofac 自动解析,无需额外的布线。Autofac 将注入一个在调用时基本上执行的委托container.Resolve<IFoo>()

在像您这样的更复杂的情况下,返回的确切具体是基于输入参数,您可以做两件事之一。首先,您可以注册一个工厂方法作为其返回值,以提供参数化解析:

builder.Register<IElement>((c, p) => {
    var dom= p.Named<IHtml>("dom");
    switch (dom.ElementType)
    {
        case "table":
            return new TableElement(dom);
        case "div":
            return new DivElement(dom);
        case "span":
            return new SpanElement(dom);
    }
    return new PassthroughElement(dom);
  });

//usage
container.Resolve<IElement>(new NamedParameter("dom", domInstance))

这不是类型安全的(domInstance 不会经过编译器检查以确保它是 IHtml),也不是很干净。相反,另一种解决方案是将工厂方法实际注册为 Func:

builder.Register<Func<IHtml, IElement>>(dom =>
{
    switch (dom.ElementType)
    {
        case "table":
            return new TableElement(dom);
        case "div":
            return new DivElement(dom);
        case "span":
            return new SpanElement(dom);
    }
    return new PassthroughElement(dom);
});


public class NeedsAnElementFactory //also registered in AutoFac
{
    protected Func<IHtml,IElement> CreateElement {get; private set;}

    //AutoFac will constructor-inject the Func you registered
    //whenever this class is resolved.
    public NeedsAnElementFactory(Func<IHtml,IElement> elementFactory)
    {
        CreateElement = elementFactory;
    }  

    public void MethodUsingElementFactory()
    {
        IHtml domInstance = GetTheDOM();

        var element = CreateElement(domInstance);

        //the next line won't compile;
        //the factory method is strongly typed to IHtml
        var element2 = CreateElement("foo");
    }
}

如果您想将代码保留在 ElementFactory 中而不是将其放入 Autofac 模块中,您可以将工厂方法设为静态并注册它(这在您的情况下特别有效,因为您的工厂方法很容易被设为静态):

public class ElementFactory
{
    public static IElement Create(IHtml dom)
    {
        switch (dom.ElementType)
        {
            case "table":
                return new TableElement(dom);
            case "div":
                return new DivElement(dom);
            case "span":
                return new SpanElement(dom);
        }
        return new PassthroughElement(dom);
    }
}

...

builder.Register<Func<IHtml, IElement>>(ElementFactory.Create);
于 2013-08-21T20:56:44.520 回答