设计模式 - 工厂模式

EN
EN
2023-09-01 / 0 评论 / 23 阅读 / 正在检测是否收录...

简单工厂模式(Simple Factory Pattern)

简单工厂模式是指由一个工厂类根据传入的参数决定创建哪一种产品类的实例。
这一模式虽然简单,但它违背了开放封闭原则,因为每添加一个新产品就需要修改工厂类逻辑代码。

工厂方法模式(Factory Method Pattern)

工厂方法模式是指将具体产品的创建延迟到具体的工厂子类中进行,由抽象工厂定义产品对象的创建接口,而具体工厂负责产生具体的产品。

抽象工厂模式(Abstract Factory Pattern)

抽象工厂模式是指针对一系列相关或相互依赖的产品组成的一个产品族,提供一个统一的接口来创建这些产品。抽象工厂模式可以用于创建一组具有同样约束的产品,同样,这一模式也存在着开放封闭原则问题,新增产品可能导致所有相关工厂类的修改。

普通的实现

interface Car {
    public function getModel();
}

class BMW implements Car{
    public function getModel() {
        return "BMW";
    }
}

class Audi implements Car{
    public function getModel() {
        return "Audi";
    }
}

class CarFactory {
    public static function createCar($type) {
        switch ($type) {
            case "BMW":
                return new BMW();
            case "Audi":
                return new Audi();
            default:
                throw new Exception("Invalid car type specified.");
        }
    }
}

$car1 = CarFactory::createCar("BMW");
echo $car1->getModel(); // BMW

$car2 = CarFactory::createCar("Audi");
echo $car2->getModel(); // Audi

在上面的代码中,我们定义了两个汽车类,BMW和Audi。
我们还定义了一个CarFactory类,该类根据传递给它的参数返回所需的汽车类的实例。
我们可以使用这个工厂类轻松创建不同类型的汽车对象。
在这个例子中,我们使用工厂模式创建了一个BMW实例和一个Audi实例。

更加优雅的实现

但是上面的代码用到了Switch显然不太优雅,我们可以借助PHP中的反射更好的实现工厂模式。反射可以为我们提供在运行时动态加载类的机制,这意味着我们可以使用反射来自动获取类名并实例化对象,而不需要在工厂类中显式指定要创建的类的名称。请看下面的代码。

<?php
// 创建一个工厂类
class CarFactory {
    public static function createCar($carType) {
        $className = ucfirst($carType).'Car'; // 构建类名
        if(class_exists($className)) { // 如果类存在
            $reflectionClass = new ReflectionClass($className); // 创建反射类
            return $reflectionClass->newInstance(); // 返回实例化后的对象
        }
        throw new Exception('Invalid car type.'); // 如果类不存在则抛出异常
    }
}

// 定义两个车型的类
class BenzCar {}
class BMWCar {}

// 调用工厂方法创建对象
$benz = CarFactory::createCar('benz');
$bmw = CarFactory::createCar('bmw');

上述代码中,我们使用ReflectionClass类来反射获取类,然后使用newInstance方法实例化对象。这样,当有新的车型类被添加到系统中时,我们无需修改工厂类的代码,仍然可以使用相同的方法进行实例化。

0

评论 (0)

取消