简介

抽象工厂模式(Abstract Factory Pattern): 抽象工厂模式提供一个接口,用于创建一系列相关或相互依赖的对象。客户端通过使用抽象工厂及其产品接口来创建对象,从而将客户端与具体的产品实现解耦。

优点

  • 客户端通过抽象工厂接口来创建对象,可以轻松替换不同的具体工厂,实现不同产品族的切换。
  • 抽象工厂保证了一系列相关产品的一致性,符合依赖倒置原则。

缺点

  • 增加新的产品族比较困难,需要修改抽象工厂的接口和所有具体工厂的实现。

应用场景

  • 对象的创建逻辑复杂:当对象的创建过程涉及到复杂的逻辑判断、依赖关系或者其他复杂的操作时,可以使用工厂模式将创建过程封装起来,简化客户端的代码。
  • 需要灵活地扩展产品族:如果需要在系统中添加新的产品族,而且这些产品族之间存在一定的关联关系,可以使用抽象工厂模式来管理和创建这些相关的产品。
  • 需要在运行时动态决定创建哪个具体对象:通过工厂模式,可以根据不同的条件或者配置在运行时动态地选择创建具体的对象,提供了更大的灵活性。

具体的场景

  • GUI控件库:在图形用户界面的控件库中,可以使用工厂模式来创建不同类型的控件,例如按钮、文本框、下拉框等。通过工厂方法或抽象工厂模式,可以根据需要创建不同风格或者主题的控件。
  • 数据库访问层:在数据库访问层中,可以使用工厂模式来创建不同类型的数据库连接对象,例如MySQL连接、Oracle连接等。根据配置或者运行时的参数,选择创建不同的数据库连接对象。
  • 日志记录器:在日志记录器的设计中,可以使用工厂模式来创建不同类型的日志记录器,例如文件日志、数据库日志、网络日志等。客户端可以通过工厂方法来获取适合自己需求的日志记录器对象。
  • 加密算法库:在加密算法库中,可以使用工厂模式来创建不同类型的加密算法对象,例如对称加密算法、非对称加密算法等。根据需要选择合适的加密算法对象进行数据加密和解密操作。

实现

案例:我们想通过抽象工厂模式实现一个简单的ORM,只有创建数据库连接和命令的功能,要做到在不同的数据库间切换而不需要修改代码。上层应用中,不管是使用哪种数据库,只需要更改工厂实例即可,要考虑代码的灵活性和可维护性。

  • 抽象工厂接口:
/// <summary>
/// 抽象工厂接口
/// </summary>
public interface IDbFactory
{
    IDbConnection CreateConnection();

    IDbCommand CreateCommand();
}
  • 实现SqlServer的工厂类:
/// <summary>
/// 针对 SQL Server 的工厂类
/// </summary>
public class SqlServerDbFactory : IDbFactory
{
    public IDbCommand CreateCommand()
    {
        return new SqlCommand();
    }

    public IDbConnection CreateConnection()
    {
        return new SqlConnection();
    }
}
  • 实现MySql的工厂类:
 /// <summary>
 /// 针对 MySQL 的工厂类
 /// </summary>
 public class MySqlDbFactory : IDbFactory
 {
     public IDbCommand CreateCommand()
     {
         return new MySqlCommand();
     }

     public IDbConnection CreateConnection()
     {
         return new MySqlConnection();
     }
 }
  • 使用抽象工厂创建数据库连接和命令:
/// <summary>
/// 使用抽象工厂创建数据库连接和命令
/// </summary>
public class DataAccess
{
    private readonly IDbFactory _dbFactory;

    public DataAccess(IDbFactory dbFactory)
    {
        _dbFactory = dbFactory;
    }

    public void ExecuteCommand()
    {
        using (var connection = _dbFactory.CreateConnection())
        {
            connection.ConnectionString = "Your Connection String";
            using (var command = _dbFactory.CreateCommand())
            {
                command.Connection = connection;
                command.CommandText = "Your SQL Command";
                connection.Open();
                command.ExecuteNonQuery();
            }
        }
    }
}
  • 上层应用调用:
{
    // 假设我们要使用 SQL Server
    IDbFactory dbFactory = new SqlServerDbFactory();
    DataAccess dataAccess = new DataAccess(dbFactory);
    dataAccess.ExecuteCommand();
}

{
    // 假设我们要使用 MySql
    IDbFactory dbFactory = new MySqlDbFactory();
    DataAccess dataAccess = new DataAccess(dbFactory);
    dataAccess.ExecuteCommand();
}

总结

总体而言,简单工厂模式适用于只有一个工厂类负责创建所有产品的场景;工厂方法模式适用于每个产品对应一个具体工厂的场景;抽象工厂模式适用于需要创建一系列相关产品的场景。根据具体的需求和系统设计的复杂度,选择适合的工厂模式可以提高代码的灵活性和可维护性。

04-06 05:51