🔥码云GVP开源项目 12k star Uniapp+ElementUI 功能强大 支持多语言、二开方便! 广告
#### PHP 设计模式系列 —— 工厂方法模式(Factory Method) Posted on 2015年12月11日 by 学院君 * * * * * 1、模式定义 定义一个创建对象的接口,但是让子类去实例化具体类。工厂方法模式让类的实例化延迟到子类中。 2、问题引出 框架需要为多个应用提供标准化的架构模型,同时也要允许独立应用定义自己的域对象并对其进行实例化。 3、解决办法 工厂方法以模板方法的方式创建对象来解决上述问题。父类定义所有标准通用行为,然后将创建细节放到子类中实现并输出给客户端。 人们通常使用工厂模式作为创建对象的标准方式,但是在这些情况下不必使用工厂方法:实例化的类永远不会改变;或者实例化发生在子类可以轻易覆盖的操作中(比如初始化)。 4、UML类图 ![](https://box.kancloud.cn/3b98bc5977ee1ba7c22753e7e1c8073d_825x246.png) 5、示例代码 FactoryMethod.php ~~~ <?php namespace DesignPatterns\Creational\FactoryMethod; /** * 工厂方法抽象类 */ abstract class FactoryMethod { const CHEAP = 1; const FAST = 2; /** * 子类必须实现该方法 * * @param string $type a generic type * * @return VehicleInterface a new vehicle */ abstract protected function createVehicle($type); /** * 创建新的车辆 * * @param int $type * * @return VehicleInterface a new vehicle */ public function create($type) { $obj = $this->createVehicle($type); $obj->setColor("#f00"); return $obj; } } ~~~ VehicleInterface.php ~~~ <?php namespace DesignPatterns\Creational\FactoryMethod; /** * VehicleInterface是车辆接口 */ interface VehicleInterface { /** * 设置车的颜色 * * @param string $rgb */ public function setColor($rgb); } ~~~ Porsche.php ~~~ <?php namespace DesignPatterns\Creational\FactoryMethod; /** * Porsche(保时捷) */ class Porsche implements VehicleInterface { /** * @var string */ protected $color; /** * @param string $rgb */ public function setColor($rgb) { $this->color = $rgb; } /** * 尽管只有奔驰汽车挂有AMG品牌,这里我们提供一个空方法仅作代码示例 */ public function addTuningAMG() { } } ~~~ Bicycle.php ~~~ <?php namespace DesignPatterns\Creational\FactoryMethod; /** * Bicycle(自行车) */ class Bicycle implements VehicleInterface { /** * @var string */ protected $color; /** * 设置自行车的颜色 * * @param string $rgb */ public function setColor($rgb) { $this->color = $rgb; } } ~~~ Ferrari.php ~~~ <?php namespace DesignPatterns\Creational\FactoryMethod; /** * Ferrari(法拉利) */ class Ferrari implements VehicleInterface { /** * @var string */ protected $color; /** * @param string $rgb */ public function setColor($rgb) { $this->color = $rgb; } } ~~~ 6、测试代码 Tests/FactoryMethodTest.php ~~~ <?php namespace DesignPatterns\Creational\FactoryMethod\Tests; use DesignPatterns\Creational\FactoryMethod\FactoryMethod; use DesignPatterns\Creational\FactoryMethod\GermanFactory; use DesignPatterns\Creational\FactoryMethod\ItalianFactory; /** * FactoryMethodTest用于测试工厂方法模式 */ class FactoryMethodTest extends \PHPUnit_Framework_TestCase { protected $type = array( FactoryMethod::CHEAP, FactoryMethod::FAST ); public function getShop() { return array( array(new GermanFactory()), array(new ItalianFactory()) ); } /** * @dataProvider getShop */ public function testCreation(FactoryMethod $shop) { // 该方法扮演客户端角色,我们不关心什么工厂,我们只知道可以可以用它来造车 foreach ($this->type as $oneType) { $vehicle = $shop->create($oneType); $this->assertInstanceOf('DesignPatterns\Creational\FactoryMethod\VehicleInterface', $vehicle); } } /** * @dataProvider getShop * @expectedException \InvalidArgumentException * @expectedExceptionMessage spaceship is not a valid vehicle */ public function testUnknownType(FactoryMethod $shop) { $shop->create('spaceship'); } } ~~~ 7、总结 工厂方法模式和抽象工厂模式有点类似,但也有不同。 工厂方法针对每一种产品提供一个工厂类,通过不同的工厂实例来创建不同的产品实例,在同一等级结构中,支持增加任意产品。 抽象工厂是应对产品族概念的,比如说,每个汽车公司可能要同时生产轿车,货车,客车,那么每一个工厂都要有创建轿车,货车和客车的方法。应对产品族概念而生,增加新的产品线很容易,但是无法增加新的产品。