抽象工廠模式(Abstact Factory)是一種常見的軟件設計模式。該模式為一個產品族提供了統一的創建接口。當需要這個產品族的某一系列的時候,可以為此系列的產品族創建一個 具體的工廠類。
【意圖】
抽象工廠模式提供一個創建一系統相關或相互依賴對象的接口,而無需指定它們具體的類【GOF95】
【抽象工廠模式結構圖】
【抽象工廠模式中主要角色】
抽象工廠(Abstract Factory)角色:它聲明一個創建抽象產品對象的接口。通常以接口或抽象類實現,所有的具體工廠類必須實現這個接口或繼承這個類。
具體工廠(Concrete Factory)角色:實現創建產品對象的操作。客戶端直接調用這個角色創建產品的實例。這個角色包含有選擇合適的產品對象的邏輯。通常使用具體類實現。
抽象產品(Abstract Product)角色:聲明一類產品的接口。它是工廠方法模式所創建的對象的父類,或它們共同擁有的接口。
具體產品(Concrete Product)角色:實現抽象產品角色所定義的接口,定義一個將被相應的具體工廠創建的產品對象。其內部包含了應用程序的業務邏輯。
【抽象工廠模式的優缺點】
抽象工廠模式的優點:
1、分離了具體的類
2、使增加或替換產品族變得容易
3、有利於產品的一致性
抽象工廠模式的缺點: 難以支持新種類的產品。這是因為AbstractFactory接口確定了可以被創建的產品集合。支持新各類的產品就需要擴展訪工廠接口,從而導致 AbstractFactory類及其所有子類的改變。
抽象工廠就是以一種傾斜的方式支持增加新的產品中,它為新產品族的增加提供了方便,而不能為新的產品等級結構的增加提供這樣的方便。
【抽象工廠模式適用場景】
以下情況應當使用抽象工廠模式:
1、一個系統不應當依賴於產品類實例如何被創建、組合和表達的細節,這對於所有形態的工廠模式都是重要的。
2、這個系統的產品有多於一個的產品族,而系統只消費其中某一族的產品。
3、 同屬於同一個產品族的產品是在一起使用的,這一約束必須在系統的設計中體現出來。
4、系統提供一個產品類的庫,所有的產品以同樣的接口出現,從而使用客戶端不依賴於實現
【Java與模式189頁】
Abstract Factory模式的幾個要點:
1、如果沒有應對“多系列對象構建”的需求變化,則沒有必要使用Abstract Factory模式。
2、“系列對象”指的是這項對象之間有相互依賴、或作用的關系。
3、Abstract Factory模式主要在於應對“新系列”的需求變動。缺點是難以應對
“新對象”的需求變動。這一點應該注意,就像前面說的,如果我們現在要在加入
其他系列的類,代碼的改動會很大。
4、Abstract Factory模式經常和Factory Method模式共同組合來應對
“對象創建”的需求變化。
抽象工廠中的增加
1. 在產品等級結構的數目不變的情況下,增加新的產品族,就意味著在每一個產品等級結構中增加一個(或者多個)新的具體 (或者抽象和具體)產品角色。 由於工廠等級結構是與產品等級結構平行的登記機構,因此,當產品等級結構有所調整時, 需要將工廠等級結構做相應的調整。現在產品等級結構中出現了新的元素,因此, 需要向工廠等級結構中加入相應的新元素就可以了。 換言之,設計師只需要向系統中加入新的具體工廠類就可以了,沒有必要修改已 有的工廠角色或者產品角色。因此,在系統中的產品族增加時,抽象工廠模式是支持“開-閉”原則的。
2. 在產品族的數目不變的情況下,增加新的產品等級結構。換言之,所有的產品等級結構 中的產品數目不會改變,但是現在多出一個與現有的產品等級結構平行的新的產品等級結構。 要做到這一點,就需要修改所有的工廠角色,給每一個工廠類都增加一個新的工廠方法, 而這顯然是違背“開–閉”原則的。換言之,對於產品等級結構的增加,抽象工廠模式是不支持“開–閉”原則的。
綜合起來,我們可以知道,在已有的抽象產品中添加其具體產品,支持“開—閉原則”, 然而在添加其抽象產品時,確不支持“開—閉”原則。抽象工廠模式以一種傾斜的 方式支持增加新的產品,它為新產品族的增加提供方便,而不能為新的產品等級 結構的增加提供這樣的方便。
【抽象工廠模式與其它模式】
單例模式(singleton模式):具體工廠類可以設計成單例類,由於工廠通常有一個就可以,因此具體工廠子類一般都實現為一個Singleton。
工廠方法模式(factory method模式):抽象工廠創建產品的方法定義為工廠方法。
原型模式(prototype模式):如果有多個可能的產品系列,具體的工廠也可以使用原型模式,具體工廠使用產品系列中。
每一個產品的原型進行實例化並且通過復制它的原型來創建新的產品。
【抽象工廠模式PHP示例】
復制代碼 代碼如下:
<?php
/**
* 抽象工廠模式 2010-05-28 sz
* @author phppan.p#gmail.com
* @package design pattern
*/
/**
* 抽象工廠
*/
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();
}
}
/**
* 抽象產品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;
}
}
/**
* 客戶端
*/
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();
?>