简单工厂、工厂方法、抽象工厂的PHP代码区别?

张开发
2026/4/4 18:40:44 15 分钟阅读
简单工厂、工厂方法、抽象工厂的PHP代码区别?
这三个模式名字很像但解决的问题层级和代码结构完全不同。简单工厂 (Simple Factory)一个类包办所有创建逻辑违反开闭原则。工厂方法 (Factory Method)每个产品对应一个工厂子类针对一个产品等级。抽象工厂 (Abstract Factory)每个系列对应一个工厂类能生产一族相关产品针对多个产品等级。以下是用 PHP 代码进行的直观对比。1. 简单工厂 (Simple Factory)核心特征静态方法 Switch/If-else。缺点每增加一个新产品必须修改工厂类的代码违反开闭原则 OCP。// --- 产品接口 ---interfaceCar{publicfunctiondrive():string;}classBenzimplementsCar{publicfunctiondrive():string{returnDriving Benz;}}classBmwimplementsCar{publicfunctiondrive():string{returnDriving Bmw;}}// --- 简单工厂 (God Class) ---classSimpleCarFactory{// 缺点每次加新车都要改这里的 switchpublicstaticfunctioncreate(string$type):Car{if($typebenz){returnnewBenz();}elseif($typebmw){returnnewBmw();}thrownewException(Invalid car type);}}// --- 调用 ---$carSimpleCarFactory::create(benz);echo$car-drive();2. 工厂方法 (Factory Method)核心特征定义工厂接口让子类决定实例化哪个类。适用只有一个产品维度比如只造车不造轮胎。优点新增产品只需新增一个工厂类无需修改旧代码符合 OCP。// --- 产品接口 (同上) ---// interface Car { ... }// class Benz implements Car { ... }// class Bmw implements Car { ... }// --- 工厂接口 ---interfaceCarFactory{publicfunctionproduce():Car;}// --- 具体工厂每个工厂只负责一种车 ---classBenzFactoryimplementsCarFactory{publicfunctionproduce():Car{returnnewBenz();}}classBmwFactoryimplementsCarFactory{publicfunctionproduce():Car{returnnewBmw();}}// --- 调用 ---$factorynewBenzFactory();// 想换车换个工厂对象即可$car$factory-produce();echo$car-drive();3. 抽象工厂 (Abstract Factory)核心特征针对“产品族”。一个工厂能生产一系列相关产品如宝马车 宝马轮胎 宝马引擎。适用有多个产品维度且需要保证它们属于同一个系列不能混用。// --- 产品族 A车 ---interfaceCar{publicfunctiondrive():string;}classBenzCarimplementsCar{publicfunctiondrive():string{returnBenz Car;}}classBmwCarimplementsCar{publicfunctiondrive():string{returnBmw Car;}}// --- 产品族 B轮胎 (新增维度) ---interfaceTire{publicfunctionroll():string;}classBenzTireimplementsTire{publicfunctionroll():string{returnBenz Tire;}}classBmwTireimplementsTire{publicfunctionroll():string{returnBmw Tire;}}// --- 抽象工厂接口定义能生产一族产品 ---interfaceVehicleFactory{publicfunctioncreateCar():Car;publicfunctioncreateTire():Tire;// 注意能产多种东西}// --- 具体工厂代表一个完整的品牌系列 ---classBenzFactoryimplementsVehicleFactory{publicfunctioncreateCar():Car{returnnewBenzCar();}publicfunctioncreateTire():Tire{returnnewBenzTire();}}classBmwFactoryimplementsVehicleFactory{publicfunctioncreateCar():Car{returnnewBmwCar();}publicfunctioncreateTire():Tire{returnnewBmwTire();}}// --- 调用 ---$factorynewBmwFactory();// 选定一个系列宝马系$car$factory-createCar();// 产出宝马车$tire$factory-createTire();// 产出宝马轮胎// 保证了一致性不会出现“宝马车 奔驰轮胎”的尴尬组合echo$car-drive(). with .$tire-roll(); 三者核心区别对比表特性简单工厂工厂方法抽象工厂核心结构1 个工厂类 if/switch1 个工厂接口 N 个工厂子类1 个工厂接口 (含多个方法) N 个工厂子类扩展新产品修改现有工厂代码 (违反 OCP)新增一个工厂类 (符合 OCP)新增一个工厂类 (符合 OCP)扩展新产品族不适用不适用 (只能扩单品)新增一组具体产品类 修改工厂接口 (违反 OCP)关注点怎么创建(集中管理)创建哪一个(单一产品)创建哪一族(配套产品)代码量最少中等 (类多)最多 (接口多、类多)PHP 场景小型脚本、快速原型支付网关、日志驱动 (单维度)云服务商 SDK (AWS vs Aliyun)、UI 主题 (Win vs Mac) 一句话总结选择策略如果只有一两种产品且以后不太会变 -简单工厂(甚至直接new)。如果产品会经常增加但只是单一类型的变化 (如只增加新的支付方式) -工厂方法。如果产品是成套出现的且需要严格保证系列一致性(如换了数据库驱动缓存驱动也要跟着换) -抽象工厂。在现代 PHP 框架如 Laravel中依赖注入容器 (DI Container)实际上接管了这些工厂模式的职责你通常只需要配置绑定关系而不需要手写这么多工厂类。但理解它们的区别有助于你设计解耦的架构。

更多文章