GOF—抽象工厂模式

158 阅读2分钟

抽象工厂模式是一种为访问类提供一个创建一组相关或相互依赖对象的接口,且访问类无须指定所要产品的具体类就能得到同族的不同等级的产品的模式结构。

抽象工厂模式是工厂方法模式的升级版本,工厂方法模式只生成一个等级的产品,抽象工厂模式可生产多个等级的产品。

  1. 抽象工厂:提供了创建产品的接口,包含多个创建产品的方法,可以创建多个不同等级的产品
  2. 具体工厂:主要是实现抽象工厂中的多个抽象方法,完成具体产品的创建。
  3. 抽象产品:定义产品的规范,描述产品的主要特性和功能,抽象工厂模式有多个抽象产品
  4. 具体产品:实现了抽象产品角色所定义的接口,由具体工厂来来创建,同具体工厂之间是多对一的关系 抽象工厂
public interface DessertFactory{
    Coffee createCoffee();
    
    Dessert createDessert();
}

具体工厂(美式风味)

public class AmericanFactory implements DessertFactory{
    @Override
    publice Coffee createCoffee(){
        return new AmericanCoffee();
    }
    
    @Override
    publice Dessert createDessert(){
        return new MatchaMousse();
    }
}

具体工厂(意大利风味)

public class ItalyFactory implements DessertFactory{
    @Override
    publice Coffee createCoffee(){
        return new LatteCoffee();
    }
    
    @Override
    publice Dessert createDessert(){
        return new Tiramisu();
    }
}

抽象产品(咖啡)

public abstract class Coffee{
    public abstract String getName();
    
    public void addMilk(){
    }
    
    public void addSugar(){
    }
}

抽象产品(甜点)

public abstract class Dessert{
    public abstract void show();
}

具体产品(美式咖啡)

public class AmericanCoffee extends Coffee{
    @Override
    public String getName(){
        return "美式咖啡";
    }
}

具体产品(拿铁咖啡)

public class LatteCoffee extends Coffee{
    @Override
    public String getName(){
        return "拿铁咖啡";
    }
}

具体产品(抹茶慕斯)

public class MatchaMousse extends Dessert{
    @Override
    public void show(){
        System.out.println("抹茶慕斯")
    }
}

具体产品(提拉米苏)

public class Tiramisu extends Dessert{
    @Override
    public void show(){
        System.out.println("提拉米苏");
    }
}

客户调用(咖啡店)

public class CoffeeStore{
    public static void main(String[] args) {
        //创建的是意大利风味甜品工厂对象
        ItalyDessertFactory factory = new ItalyDessertFactory();
       
        //获取拿铁咖啡和提拉米苏甜品
        Coffee coffee = factory.createCoffee();
        Dessert dessert = factory.createDessert();

        System.out.println(coffee.getName());
        dessert.show();
    }
}

优点:当一个产品族中的多个对象被设计成一起工作时,它能保证客户端始终只使用一个产品族中的对象。

缺点:当产品族中需要增加一个新的产品时,所有的工厂类都需要进行修改。