假设我们有一个为我们创建一些产品的抽象工厂。我们知道抽象工厂可以为我们提供产品的一些特定子类,但我们不想检查类型(这是这种模式的主要原因)。现在我们需要为每种类型的对象创建一个特定的视图,我们如何在不知道具体类型的情况下做到这一点呢?同一个工厂应该创建不同的视图吗?
更新:我创建了一个github 存储库来尝试所有不同的方法。
假设我们有一个为我们创建一些产品的抽象工厂。我们知道抽象工厂可以为我们提供产品的一些特定子类,但我们不想检查类型(这是这种模式的主要原因)。现在我们需要为每种类型的对象创建一个特定的视图,我们如何在不知道具体类型的情况下做到这一点呢?同一个工厂应该创建不同的视图吗?
更新:我创建了一个github 存储库来尝试所有不同的方法。
对于这个问题,我们可以看看抽象工厂模式是如何在 ADO.NET 中实现的。
我们有一个名为 的抽象工厂DbProviderFactory
。该工厂有多种实现方式,例如,SqlClientFactory
等。MySqlClientFactory
OracleClientFactory
对于这个问题,产品是与数据库相关的对象,如连接、命令、数据适配器等。
起初,我们的抽象工厂给了我们一个连接(产品)。它可能是MySqlConnection
或OracleConnection
。我们唯一知道的是,它是一个DbConnection
.
DbProviderFactory factory = ...
DbConnection conn = factory.CreateConnection();
现在我们需要创建一个可以与此连接一起使用的命令对象(视图)。
DbCommand cmd = conn.CreateCommand();
如您所见,命令(视图)是由连接(产品)而不是抽象工厂创建的。
但这并不完全正确……
实际发生的事情隐藏在实现细节中。当抽象工厂创建连接时,它会将自己传递给连接。当我们请求命令时,连接通过提供的抽象工厂创建命令。
所以,如果我们回到你的问题,实现可以是这样的。
interface IView {
IProduct Product { get; set; }
void Render();
}
interface IProduct {
IView CreateView();
}
interface IAbstractFactory {
IProduct CreateProduct();
IView CreateView();
}
class View1 : IView {
public IProduct Product { get; set; }
public void Render() {
Product1 p1 = (Product1)Product;
// Do product1 specific rendering here
}
}
class Product1 : IProduct {
private IAbstractFactory factory;
public Product1(IAbstractFactory factory) {
this.factory = factory;
}
public IView CreateView() {
IView view = factory.CreateView();
view.Product = this;
return this;
}
}
class Factory1 : IAbstractFactory {
IProduct CreateProduct() {
return new Product1(this);
}
IView CreateView() {
return new View1();
}
}