设计模式之工厂模式

文章目录

简单工厂模式

没有工厂模式时,客户端要自己 new 对象:

java 复制代码
Vehicle car = new Car();
Vehicle bike = new Bike();

这带来两个问题:

复制代码
1. 客户端需要知道所有产品的具体类名
2. 创建逻辑散落在各处,想改就得到处找

核心思想

把创建对象的逻辑集中到一个工厂类里,客户端不直接 new 对象。

实现

java 复制代码
interface Vehicle {
      void drive();
  }

class Car implements Vehicle {
      @Override
      public void drive() { System.out.println("开车"); }
  }

class Bike implements Vehicle {
      @Override
      public void drive() { System.out.println("骑自行车"); }
  }

class Truck implements Vehicle {
      @Override
      public void drive() { System.out.println("开卡车"); }
  }

  // 简单工厂
class SimpleVehicleFactory {
      public static Vehicle create(String type) {
          if ("car".equals(type))   return new Car();
          if ("bike".equals(type))  return new Bike();
          if ("truck".equals(type)) return new Truck();
          throw new IllegalArgumentException("Unknown type: " + type);
      }
  }

使用

java 复制代码
Vehicle car = SimpleVehicleFactory.create("car");
car.drive(); // 开车

Vehicle bike = SimpleVehicleFactory.create("bike");
bike.drive(); // 骑自行车

缺点

每次新增产品(比如 ElectricCar),都必须修改 SimpleVehicleFactory 的 if-else,违背开闭原则。

工厂方法模式

简单工厂的核心矛盾在于:所有创建逻辑都压在一个类里,新增产品必然要动这个类。

工厂方法的解决思路很直接------把那个大 if-else 拆开,每个分支变成一个独立的工厂子类。

核心思想

定义一个创建对象的接口,但把实例化哪个类的决定权延迟到子类。

实现

第一步:产品接口不变

java 复制代码
interface Vehicle {
      void drive();
  }

class Car implements Vehicle {
      @Override
      public void drive() { System.out.println("开车"); }
  }

class Bike implements Vehicle {
      @Override
      public void drive() { System.out.println("骑自行车"); }
  }

第二步:定义抽象工厂

java 复制代码
interface VehicleFactory {
      Vehicle createVehicle(); // 工厂方法,创建什么由子类决定
  }

第三步:每种产品对应一个具体工厂

java 复制代码
class CarFactory implements VehicleFactory {
      @Override
      public Vehicle createVehicle() { return new Car(); }
  }

class BikeFactory implements VehicleFactory {
      @Override
      public Vehicle createVehicle() { return new Bike(); }
  }

使用

java 复制代码
VehicleFactory factory = new CarFactory();
Vehicle car = factory.createVehicle();
car.drive(); // 开车

factory = new BikeFactory();
Vehicle bike = factory.createVehicle();
bike.drive(); // 骑自行车

验证

新增电动车,零修改旧代码

java 复制代码
 // 新增产品
class ElectricCar implements Vehicle {
      @Override
      public void drive() { System.out.println("开电动车"); }
  }

  // 新增工厂,完全不动任何现有类
class ElectricCarFactory implements VehicleFactory {
      @Override
      public Vehicle createVehicle() { return new ElectricCar(); }
  }

缺点

  • 类数量膨胀:产品每增加一种,就要同步新增一个工厂类
  • 只能生产单一产品:一个工厂只对应一种产品,无法保证产品之间的关联性

抽象工厂模式

工厂方法解决了扩展性问题,但有个新场景它处理不好:

假设现在有 A 品牌和 B 品牌,每个品牌都生产汽车和自行车。客户端想要一套 A品牌的产品,用工厂方法需要分别调用 ACarFactory 和 ABikeFactory,无法保证拿到的产品是同一品牌的(代码层面没有约束机制,靠的是开发者自觉,一旦粗心就会混用)。

核心思想

一个工厂不再生产单一产品,而是生产一套相关联的产品族,保证客户端只拿到一套配套的产品。

实现

java 复制代码
  // 产品接口
interface Car  { void drive(); }
interface Bike { void ride();  }

  // A 品牌产品
class ACar implements Car {
      @Override public void drive() { System.out.println("A 品牌轿车"); }
  }
class ABike implements Bike {
      @Override public void ride()  { System.out.println("A 品牌自行车"); }
  }

  // B 品牌产品
class BCar implements Car {
      @Override public void drive() { System.out.println("B 品牌轿车"); }
  }
class BBike implements Bike {
      @Override public void ride()  { System.out.println("B 品牌自行车"); }
  }

  // 抽象工厂:生产一套产品
interface Factory {
      Car  createCar();
      Bike createBike();
  }

  // A 品牌工厂,只生产 A 品牌的一套产品
class AFactory implements Factory {
      @Override public Car  createCar()  { return new ACar();  }
      @Override public Bike createBike() { return new ABike(); }
  }

  // B 品牌工厂,只生产 B 品牌的一套产品
class BFactory implements Factory {
      @Override public Car  createCar()  { return new BCar();  }
      @Override public Bike createBike() { return new BBike(); }
  }

使用

java 复制代码
  // 切换品牌只需换一行
Factory factory = new AFactory();
Car  car  = factory.createCar();   // A 品牌轿车
Bike bike = factory.createBike();  // A 品牌自行车,品牌一致 ✅

  // 换成 B 品牌
factory = new BFactory();
car  = factory.createCar();        // B 品牌轿车
bike = factory.createBike();       // B 品牌自行车,品牌一致 ✅
相关推荐
Flittly7 小时前
【AgentScope Java新手村系列】(16)从RAG到多路检索
java·spring boot·spring
小兔崽子去哪了7 小时前
Java 生成二维码解决方案
java·后端
人活一口气12 小时前
从JVM调优到MCP协议:Java全栈技术体系深度总结与企业级架构实践
java·spring boot
NE_STOP13 小时前
Vibe Coding -- 完整项目案例实操
java
荣码13 小时前
GraphRAG:普通RAG只能回答"点"的问题,我踩了4个坑才搞懂
java·python
SimonKing13 小时前
Google第三方授权登录
java·后端·程序员
明月光81814 小时前
从一行 @Builder 说起:重新拾起 Java 的 Lombok、注解与 Builder 模式
java
考虑考虑1 天前
Mybatis实现批量插入
java·后端·mybatis
咖啡八杯1 天前
GoF设计模式——中介者模式
java·后端·spring·设计模式
青石路1 天前
记一次多JDK版本问题的排查,一坑套一坑,差点没爬上来
java