创建型模式 - 抽象工厂模式

0x01 前言

  继上一篇文章所述,抽象工厂模式(Abstract Factory Pattern)是围绕一个超级工厂创建其他工厂。该超级工厂又称为其他工厂的工厂。这种类型的设计模式属于创建型模式,它提供了一种创建对象的最佳方式。

  在抽象工厂模式中,接口是负责创建一个相关对象的工厂,不需要显式指定它们的类。每个生成的工厂都能按照工厂模式提供对象。

0x02 简介

意图:提供一个创建一系列相关或相互依赖对象的接口,而无需指定它们具体的类。

主要解决:主要解决接口选择的问题。

何时使用:系统的产品有多于一个的产品族,而系统只消费其中某一族的产品。

如何解决:在一个产品族里面,定义多个产品。

关键代码:在一个工厂里聚合多个同类产品。

应用实例:工作了,为了参加一些聚会,肯定有两套或多套衣服吧,比如说有商务装(成套,一系列具体产品)、时尚装(成套,一系列具体产品),甚至对于一个家庭来说,可能有商务女装、商务男装、时尚女装、时尚男装,这些也都是成套的,即一系列具体产品。假设一种情况(现实中是不存在的,要不然,没法进入共产主义了,但有利于说明抽象工厂模式),在您的家中,某一个衣柜(具体工厂)只能存放某一种这样的衣服(成套,一系列具体产品),每次拿这种成套的衣服时也自然要从这个衣柜中取出了。用 OO 的思想(面向对象)去理解,所有的衣柜(具体工厂)都是衣柜类的(抽象工厂)某一个,而每一件成套的衣服又包括具体的上衣(某一具体产品),裤子(某一具体产品),这些具体的上衣其实也都是上衣(抽象产品),具体的裤子也都是裤子(另一个抽象产品)。

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

缺点:产品族扩展非常困难,要增加一个系列的某一产品,既要在抽象的 Creator 里加代码,又要在具体的里面加代码。

使用场景: 1、QQ 换皮肤,一整套一起换。 2、生成不同操作系统的程序。

注意事项:产品族难扩展,产品等级易扩展。

0x03 设计概述

  抽象工厂模式是工厂方法模式的升级版本,他用来创建一组相关或者相互依赖的对象。

  抽象工厂模式时代:随着客户的要求越来越高,需要定制不同组合的电脑。例如联想 Y 系列,华硕 G 系列,联想 Y 配备性能级显卡,联想 G 配备发烧级显卡。那么并不需要创办一个生产联想 Y 性能级显卡电脑的工厂、华硕 G 发烧级显卡电脑的工厂。而仅仅是抽象一个工厂,用来生产联想 Y 电脑、华硕 G 电脑、性能级显卡、发烧级显卡,然后组合就形成产品。

0x04 具体实现

抽象工厂时代
项目结构图
项目结构图.PNG
产品类

  为产品显卡创建接口。

// factory_pattern.abstract_factory.product.IGraphics
public interface IGraphics {
    void getGraphics(String series);
}

  为产品电脑创建接口。

// factory_pattern.abstract_factory.product.IComputer
public interface IComputer {
    void getComputer(String brand);
}

  拓展上述接口,采用需要组合的方式生产产品,所以将类进行细分拓展接口。

  拓展显卡接口 IGraphics ,定义性能级显卡类 PerformanceGraphics

// factory_pattern.abstract_factory.product.impl.PerformanceGraphics
public class PerformanceGraphics implements IGraphics {
    @Override
    public void getGraphics(String series) {
        System.out.println("研发一个 " + series + " 显卡");
    }
}

  拓展显卡接口 IGraphics ,定义发烧级显卡类 FeverGraphics

// factory_pattern.abstract_factory.product.impl.FeverGraphics
public class FeverGraphics implements IGraphics {
    @Override
    public void getGraphics(String series) {
        System.out.println("制造一个 " + series + " 显卡");
    }
}

  拓展电脑接口 IComputer ,定义联想电脑类 LenovoComputer

// factory_pattern.abstract_factory.product.impl.LenovoComputer
public class LenovoComputer implements IComputer {
    @Override
    public void getComputer(String brand) {
        System.out.println("制造一台 " + brand + " 笔记本电脑");
    }
}

  拓展电脑接口 IComputer ,定义华硕电脑类 AsusComputer

// factory_pattern.abstract_factory.product.impl.AsusComputer
public class AsusComputer implements IComputer {
    @Override
    public void getComputer(String brand) {
        System.out.println("研发一台 " + brand + " 台式电脑");
    }
}
工厂类

  为 Computer 和 Graphics 对象创建接口 IAbstractFactory 来获取工厂。

// factory_pattern.abstract_factory.factory.IAbstractFactory
public interface IAbstractFactory {
    IComputer createComputer(String brand);
    IGraphics createGraphics(String series);
}

  创建扩展了 IAbstractFactory 的工厂接口,基于给定的信息生成实体类的对象。

// factory_pattern.abstract_factory.factory.impl.GraphicsFactroy
public class GraphicsFactroy implements IAbstractFactory {
    @Override
    public IComputer createComputer(String brand) {
        return null;
    }
    @Override
    public IGraphics createGraphics(String series) {
        switch (series) {
            case "Performance" :
                return new PerformanceGraphics();
            case "Fever" :
                return new FeverGraphics();
            default :
                throw new IllegalArgumentException();
        }
    }
}
// factory_pattern.abstract_factory.factory.impl.ComputerFactory
public class ComputerFactory implements IAbstractFactory {
    @Override
    public IComputer createComputer(String brand) {
        switch (brand) {
            case "Lenovo" :
                return new LenovoComputer();
            case "Asus" :
                return new AsusComputer();
            default :
                throw new IllegalArgumentException();
        }
    }
    @Override
    public IGraphics createGraphics(String series) {
        return null;
    }
}

  创建一个工厂创造器 / 生成器类,通过传递品牌或型号来获取工厂。

// factory_pattern.abstract_factory.factory.impl.FactoryProducer
public class FactoryProducer {
    public static IAbstractFactory getFactory(String choice){
        switch (choice) {
            case "Computer" :
                return new ComputerFactory();
            case "Graphics" :
                return new GraphicsFactroy();
            default :
                throw new IllegalArgumentException();
        }
    }
}
用户类

  使用 FactoryProducer 来获取 IAbstractFactory,通过传递类型信息来获取实体类的对象。

// factory_pattern.abstract_factory.user.ComputerTest
public class ComputerTest {
    @Test
    public void testGetLenovoPerformanceComputer() {
        IAbstractFactory computerFactory = FactoryProducer.getFactory("Computer");
        IComputer lenPC = computerFactory.createComputer("Lenovo");
        IAbstractFactory graphicsFactory = FactoryProducer.getFactory("Graphics");
        IGraphics lenPer = graphicsFactory.createGraphics("Performance");
        lenPC.getComputer("Lenovo");
        lenPer.getGraphics("Performance");
    }
    @Test
    public void testGetAsusFeverComputer() {
        IAbstractFactory computerFactory = FactoryProducer.getFactory("Computer");
        IComputer lenPC = computerFactory.createComputer("Asus");
        IAbstractFactory graphicsFactory = FactoryProducer.getFactory("Graphics");
        IGraphics lenPer = graphicsFactory.createGraphics("Fever");
        lenPC.getComputer("Asus");
        lenPer.getGraphics("Fever");
    }
}
验证输出

testGetLenovoPerformanceComputer

制造一台 Lenovo 笔记本电脑
研发一个 Performance 显卡

testGetAsusFeverComputer

研发一台 Asus 台式电脑
制造一个 Fever 显卡

0x05 总结

  无论是简单工厂,工厂方法模式,还是抽象工厂模式,他们都属于工厂模式,在形式和特点上也是极为相似的,他们的最终目的都是为了解耦。在使用时,我们不必去在意这个模式到底工厂方法模式还是抽象工厂模式,因为他们之间的演变常常是令人琢磨不透的。经常你会发现,明明使用的工厂方法模式,当新需求来临,稍加修改,加入了一个新方法后,由于类中的产品构成了不同等级结构中的产品族,它就变成抽象工厂模式了。而对于抽象工厂模式,当减少一个方法使的提供的产品不再构成产品族之后,它就演变成了工厂方法模式。

  所以,在使用工厂模式时,只需要关心降低耦合度的目的是否达到了。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容