## **抽象工厂模式**
抽象工厂模式(Abstact Factory)是一种常见的软件设计模式。该模式为一个产品族提供了统一的创建接口。当需要这个产品族的某一系列的时候,可以为此系列的产品族创建一个具体的工厂类。
**一、意图**
抽象工厂模式提供一个创建一系统相关或相互依赖对象的接口,而无需指定它们具体的类【GOF95】
**二、抽象工厂模式结构图**
![](https://img.kancloud.cn/59/d7/59d730fe9ff928a89d31cd4f7b6e24b2_414x439.png)
**三、抽象工厂模式中主要角色**
**抽象工厂(Abstract Factory)角色:**它声明一个创建抽象产品对象的接口。通常以接口或抽象类实现,所有的具体工厂类必须实现这个接口或继承这个类。
**具体工厂(Concrete Factory)角色:**实现创建产品对象的操作。客户端直接调用这个角色创建产品的实例。这个角色包含有选择合适的产品对象的逻辑。通常使用具体类实现。
**抽象产品(Abstract Product)角色:**声明一类产品的接口。它是工厂方法模式所创建的对象的父类,或它们共同拥有的接口。
**具体产品(Concrete Product)角色:**实现抽象产品角色所定义的接口,定义一个将被相应的具体工厂创建的产品对象。其内部包含了应用程序的业务逻辑。
**四、抽象工厂模式的优缺点
**抽象工厂模式的**优点**:
1、分离了具体的类
2、使增加或替换产品族变得容易
3、有利于产品的一致性
抽象工厂模式的**缺点**: 难以支持新种类的产品。这是因为AbstractFactory接口确定了可以被创建的产品集合。支持新各类的产品就需要扩展访工厂接口,从而导致AbstractFactory类及其所有子类的改变。
抽象工厂就是以一种倾斜的方式支持增加新的产品中,它为新产品族的增加提供了方便,而不能为新的产品等级结构的增加提供这样的方便。
**五、抽象工厂模式适用场景
**以下情况应当使用抽象工厂模式:
1、一个系统不应当依赖于产品类实例如何被创建、组合和表达的细节,这对于所有形态的工厂模式都是重要的。
2、这个系统的产品有多于一个的产品族,而系统只消费其中某一族的产品。
3、 同属于同一个产品族的产品是在一起使用的,这一约束必须在系统的设计中体现出来。
4、系统提供一个产品类的库,所有的产品以同样的接口出现,从而使用客户端不依赖于实现
**六、抽象工厂模式与其它模式
**单例模式(singleton模式):具体工厂类可以设计成单例类,由于工厂通常有一个就可以,因此具体工厂子类一般都实现为一个Singleton。
工厂方法模式(factory method模式):抽象工厂创建产品的方法定义为工厂方法。
原型模式(prototype模式):如果有多个可能的产品系列,具体的工厂也可以使用原型模式,具体工厂使用产品系列中
每一个产品的原型进行实例化并且通过复制它的原型来创建新的产品。
**七、抽象工厂模式PHP示例**
~~~
<?php
/**
* 抽象产品A
*/
interface AbstractProductA {
/**
* 取得产品名
*/
public function getName();
}
/**
* 抽象产品B
*/
interface AbstractProductB {
/**
* 取得产品名
*/
public function getName();
}
/**
* 具体产品A1
*/
class ProductA1 implements AbstractProductA {
private $_name;
public function __construct() {
$this->_name = 'product A1';
}
public function getName() {
return $this->_name;
}
}
/**
* 具体产品A2
*/
class ProductA2 implements AbstractProductA {
private $_name;
public function __construct() {
$this->_name = 'product A2';
}
public function getName() {
return $this->_name;
}
}
/**
* 具体产品B1
*/
class ProductB1 implements AbstractProductB {
private $_name;
public function __construct() {
$this->_name = 'product B1';
}
public function getName() {
return $this->_name;
}
}
/**
* 具体产品B2
*/
class ProductB2 implements AbstractProductB {
private $_name;
public function __construct() {
$this->_name = 'product B2';
}
public function getName() {
return $this->_name;
}
}
/**
* 抽象工厂
*/
interface AbstractFactory {
/**
* 创建等级结构为A的产品的工厂方法
*/
public function createProductA();
/**
* 创建等级结构为B的产品的工厂方法
*/
public function createProductB();
}
/**
* 具体工厂1
*/
class ConcreteFactory1 implements AbstractFactory{
public function createProductA() {
return new ProductA1();
}
public function createProductB() {
return new ProductB1();
}
}
/**
* 具体工厂2
*/
class ConcreteFactory2 implements AbstractFactory{
public function createProductA() {
return new ProductA2();
}
public function createProductB() {
return new ProductB2();
}
}
/**
* 客户端
*/
class Client {
/**
* Main program.
*/
public static function main() {
self::run(new ConcreteFactory1());
self::run(new ConcreteFactory2());
}
/**
* 调用工厂实例生成产品,输出产品名
* @param $factory AbstractFactory 工厂实例
*/
public static function run(AbstractFactory $factory) {
$productA = $factory->createProductA();
$productB = $factory->createProductB();
echo $productA->getName(), '<br />';
echo $productB->getName(), '<br />';
}
}
Client::main();
?>
~~~
- PHP篇
- 函数传值和传引用的区别
- 简述PHP的垃圾回收机制
- 简述CGI、FAST-CGI、PHP-FPM的关系
- 常见正则表达式
- 多进程写文件,如何保证都写成功
- php支持回调函数的数组函数
- MySQL篇
- MySQL的两种存储引擎区别
- 事务的四大特性
- 数据库事务隔离级别
- 什么是索引
- 索引有哪些数据结构,优缺点
- 索引的一些潜规则
- SQL的优化方案
- 简述MySQL的锁机制
- 死锁是怎么产生的?怎么解决?
- 简述MySQL的主从复制过程,延迟问题怎么解决
- 分布式事务的解决方案
- 数据库中间件MyCat
- Linux篇
- Linux常用命令
- 对日志文件的IP出现的次数进行统计,并显示次数最多的前5名
- WEB篇
- 跨域是怎么产生的,如何解决跨域
- Redis篇
- redis介绍
- redis和memcached区别
- redis的持久化方案
- 缓存穿透、击穿、雪崩、预热、更新、降级
- 网络篇
- 计算机网络体系结构
- 简述TCP的三次握手、四次挥手过程
- UDP、TCP 区别,适用场景
- HTTP常见状态码含义
- 设计模式篇
- 单例模式
- 简单工厂模式
- 抽象工厂模式
- 观察者模式
- 策略模式
- 注册模式
- 适配器模式
- 安全篇
- 跨站脚本攻击(XSS)
- 跨站点请求伪造(CSRF)
- SQL 注入
- 应用层拒绝服务攻击
- PHP安全
- 运维篇
- docker面试题
- 消息队列篇
- 架构篇
- 数据结构与算法篇