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

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 总结

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

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

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,456评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,370评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,337评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,583评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,596评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,572评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,936评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,595评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,850评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,601评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,685评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,371评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,951评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,934评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,167评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,636评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,411评论 2 342

推荐阅读更多精彩内容