如何在工厂中实现构造函数注入?

时间:2008-12-16 23:29:36

标签: c# design-patterns dependency-injection factory

当您使用工厂模式时,如何在运行时将依赖项注入构造函数?

我正在构建具有不同格式的Foos - 布尔值,数组,自由文本,矩阵等。当我们发现Foo的不同用途时,格式列表将会增长。这是我的基本核心领域:

public interface IFoo
{
    FooFormat Format { get; }
}

public class Foo : IFoo
{
    private FooFormat _format;

    internal Foo(FooFormat format)
    {
        _format = format;
    }

    public FooFormat Format { get { return _format; } }
}


public abstract class FooFormat
{
}

public class DefaultFooFormat : FooFormat
{
}

public class BooleanFooFormat : FooFormat
{
    public IList<bool> Values { get; set; }
}

public class ArrayFooFormat : FooFormat
{
    private IList<string> _values;

    public ArrayFooFormat(IList<string> values)
    {
        _values = values;
    }

    public IList<string> Values { get { return _values; } }
}

为消费者环境设计IFoo:

public abstract class FooDecorator : IFoo
{
    private IFoo _foo;

    protected FooDecorator(IFoo foo)
    {
        _foo = foo;
    }

    public FooFormat Format
    {
        get { return _foo.Format; }
    }

    protected IFoo foo
    {
        get { return _foo; }
    }
}

我不希望我的消费者直接实例化Foo,所以我强迫他们使用工厂:

public abstract class FooFactory
{
    protected IFoo Build<T>()
    {
        FooFormat format = GetFormat<T>();
        return new Foo(format);
    }

    private FooFormat GetFormat<T>()
    {
        if (typeof(T) == typeof(ArrayFooFormat)) return new ArrayFooFormat(new List<string>());
        if (typeof(T) == typeof(BooleanFooFormat)) return new BooleanFooFormat();
        return new DefaultFooFormat();
    }
}

即便如此,他们还需要从我的抽象工厂中找到一个工厂,以满足他们的特定背景。

我特意在html上下文中构建foos,如下所示:

public class HtmlFoo : FooDecorator
{
    public HtmlFoo(IFoo foo) : base(foo) { }

    public string ToHtml()
    {
        return "<div>" + this.Format.ToString() + "</div>";
    }
}


public class HtmlFooFactory : FooFactory
{
    public IFoo BuildFoo<T>()
    {
        IFoo foo = Build<T>();
        return new HtmlFoo(foo);
    }
}

public class HtmlFooConsumer
{
    public void DoSomeFoo()
    {
        var factory = new HtmlFooFactory();
        var htmlBooleanFoo = factory.BuildFoo<BooleanFooFormat>();
        var htmlArrayFoo = factory.BuildFoo<ArrayFooFormat>();
    }
}

我的问题在于我的抽象FooFactory:我总是在我的ArrayFooFormat中注入一个空值列表。我希望能够传递消费者的价值表。对于其他FooFormats,我想从使用者传递正确的构造函数参数。但是我想让公共API变得简单 - 我不想在BuildFoo()上出现一堆重载。

那么如何将自定义值列表传递到来自HtmlFooConsumer.DoSomeFoo()内部的factory.BuildFoo&lt; T&gt;()调用?任何想法,stackoverflow大师?

2 个答案:

答案 0 :(得分:2)

也许你可以在这些行中做一些事情,你的抽象FooFormat变成了IFooFormat,而泛型FooFormat提供了一个传递参数的Init方法。

然后,Build的一次重载允许您传入参数。

public interface IFooFormat
{
}

public class FooFormat<TValue> : IFooFormat
{
    private TValue _value;

    public void Init(TValue value)
    {
        _value = value;
    }

    public TValue Value
    {
        get { return _value; }
    }
}

public class ArrayFooFormat : FooFormat<IList<string>> { }

public class BooleanFooFormat : FooFormat<bool> { }

public class DefaultFooFormat : IFooFormat { }

public interface IFoo { }

public class Foo : IFoo
{
    private IFooFormat _format;

    internal Foo(IFooFormat format)
    {
        _format = format;
    }

    public IFooFormat Format { get { return _format; } }
}

public class FooFactory
{
    protected IFoo Build<TFormat, TArg>(TArg arg) where TFormat : FooFormat<TArg>, new()
    {
        TFormat format = new TFormat();
        format.Init(arg);
        return new Foo(format);
    }

    protected IFoo Build<TFormat>() where TFormat : IFooFormat, new()
    {
        return new Foo(new TFormat());
    }
}

答案 1 :(得分:0)

工厂基本上是静态变量的面向对象版本。我完全避免使用一个。您可以简单地将对象注入其构造函数,而不是强迫客户使用工厂,而不需要工厂。