理解设计模式之工厂模式

本贴最后更新于 2133 天前,其中的信息可能已经事过景迁

什么是工厂模式?

  上一篇文章写了关于创建模式之一的单例模式,这篇依然是关于创建模式的,最常用的模式之一,工厂模式。那么到底什么是工厂模式?理解工厂模式的概念很容易,光看工厂两个字就知道了,就是用来生产东西的,工厂模式,可以理解为专门用于生产对象的类,这个类不做其他的,就是生产对象。那么为什么会出现这个模式?我们直接 new 一个对象不就好了吗?还这么麻烦要专门用一个类来干这个事,不是多此一举么?其实不然,工厂模式提倡了“面向接口”编程的思想,将具体对象的创建跟业务代码解耦,让业务不用关心对象是如何产生的,只需要调用业务对象的接口即可。你想想,如果是 new 一个对象,然后再初始化这个对象的一些属性,万一哪天对象发生了变化,增加了一个需要初始化的属性,岂不是所有 new 这个对象的地方都会改一遍?这明显违背了“开闭原则”。因此工厂模式的出现,正是解决对象创建跟业务代码紧耦合这个问题。

工厂模式分类

  工厂模式就是工厂模式,干嘛还要分类呢?初学者可能对这个问题比较疑惑。为啥要分一下类呢?因为工厂这个东西,会随着产品的复杂程度而变化,跟现实世界中的一样,一个简单的工厂流水线不可能生产出种类繁多又很复杂的产品,那么随着产品的变化,工厂的实现也会跟着变化。根据以往查看过的资料来看,工厂模式一共分为三类:简单工厂、工厂方法、抽象工厂。看了一下网上的资料,与以往的总结,简单描述一下自己对这三类工厂实现的理解(只讲原理,不贴代码)。
  简单工厂 Simple Factory
  简单工厂,为啥要叫简单工厂?因为简单。我这个工厂只生产一个种类的产品,不生产其他东西,那么这个工厂就是简单工厂。例如假如你开了一个面食厂,专门做包子,生产肉包、菜包和豆沙包(如果你想买馒头,不好意思,到隔壁老王的馒头店去)。结合 UML 类图理解一下:
png
  工厂方法 Factory Method
  上面是简单工厂,只能生产一些简单的东西,我就生产这类产品,还是拿面食厂举例,面食厂开张后,生意好了,吃得人越来越多,时间久了大家觉得一个师傅做出来的口味始终是一个味道(想想你们在公司食堂吃了一年的饭菜),这时候突然有人提议,要吃换个口味的包子,但是包子还是肉包、菜包和豆沙包,这个时候怎么办?那么就轮到工厂方法出马了,工厂方法模式生产的产品还是那几样,但是实现者却不一样,实现的产品也不一样,拿面食店举例,假如之前是请老干妈做的包子,那么现在又请了一个老干爹,做口味不一样的包子。结合 UML 类图理解一下:
png
  虽然都是肉包子,但是老干妈的肉包子跟老干爹的肉包子是不一样的口味,因此工厂方法,适合同款产品有多种不同实现的场景。最简单的应用场景如数据库驱动,数据库驱动标准接口是
java.sql.Driver,Driver 生产 java.sql.Connection 对象。实现 java.sql.Driver 接口的驱动程序有跟多,例如 MySQL 的驱动与 Oracle 的驱动,两者生产的 Connection 对象也分别是基于 MySQL 数据库的实现和 Oracle 数据库的实现,但是对于业务开发人员来说,并不用关心到底是 Connection 是什么实现,我只需要定义好对应数据库的驱动程序即可。
  抽象工厂 Abstract Factory
  抽象工厂是在工厂方法之上的又一次扩展,随着产品的复杂度提高,工厂方法以及不能适应更多产品的生产,拿面食工厂举例,时间久了,大家对面食的种类产生了疑惑,为啥没有馒头?要求面食工厂能生产馒头,那么好,现在有两位师傅,一个是老干妈,一个是老干爹,从现在开始让他们生产馒头,我们工厂现在的产品种类有包子和馒头(哪天说不定还有花卷),在这种支持产品种类会有扩展的场景,就需要抽象工厂设计模式来搞定了。集合 UML 类图理解一下抽象工厂:
png
抽象工厂适合产品种类会扩展的场景,但是这种模式也有弊端,在进行产品种类扩展的同时,也会去修改已有的工厂类,添加新的产品生产接口,这个违背了开闭原则,因此要谨慎使用。

  • Java

    Java 是一种可以撰写跨平台应用软件的面向对象的程序设计语言,是由 Sun Microsystems 公司于 1995 年 5 月推出的。Java 技术具有卓越的通用性、高效性、平台移植性和安全性。

    3187 引用 • 8213 回帖
  • 设计模式

    设计模式(Design pattern)代表了最佳的实践,通常被有经验的面向对象的软件开发人员所采用。设计模式是软件开发人员在软件开发过程中面临的一般问题的解决方案。这些解决方案是众多软件开发人员经过相当长的一段时间的试验和错误总结出来的。

    200 引用 • 120 回帖 • 1 关注

相关帖子

欢迎来到这里!

我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。

注册 关于
请输入回帖内容 ...

推荐标签 标签

  • 导航

    各种网址链接、内容导航。

    40 引用 • 173 回帖 • 1 关注
  • CloudFoundry

    Cloud Foundry 是 VMware 推出的业界第一个开源 PaaS 云平台,它支持多种框架、语言、运行时环境、云平台及应用服务,使开发人员能够在几秒钟内进行应用程序的部署和扩展,无需担心任何基础架构的问题。

    5 引用 • 18 回帖 • 169 关注
  • 区块链

    区块链是分布式数据存储、点对点传输、共识机制、加密算法等计算机技术的新型应用模式。所谓共识机制是区块链系统中实现不同节点之间建立信任、获取权益的数学算法 。

    91 引用 • 751 回帖 • 4 关注
  • API

    应用程序编程接口(Application Programming Interface)是一些预先定义的函数,目的是提供应用程序与开发人员基于某软件或硬件得以访问一组例程的能力,而又无需访问源码,或理解内部工作机制的细节。

    77 引用 • 430 回帖
  • Redis

    Redis 是一个开源的使用 ANSI C 语言编写、支持网络、可基于内存亦可持久化的日志型、Key-Value 数据库,并提供多种语言的 API。从 2010 年 3 月 15 日起,Redis 的开发工作由 VMware 主持。从 2013 年 5 月开始,Redis 的开发由 Pivotal 赞助。

    286 引用 • 248 回帖 • 61 关注
  • 数据库

    据说 99% 的性能瓶颈都在数据库。

    342 引用 • 708 回帖
  • 服务

    提供一个服务绝不仅仅是简单的把硬件和软件累加在一起,它包括了服务的可靠性、服务的标准化、以及对服务的监控、维护、技术支持等。

    41 引用 • 24 回帖 • 2 关注
  • App

    App(应用程序,Application 的缩写)一般指手机软件。

    91 引用 • 384 回帖
  • 互联网

    互联网(Internet),又称网际网络,或音译因特网、英特网。互联网始于 1969 年美国的阿帕网,是网络与网络之间所串连成的庞大网络,这些网络以一组通用的协议相连,形成逻辑上的单一巨大国际网络。

    98 引用 • 344 回帖 • 1 关注
  • Vditor

    Vditor 是一款浏览器端的 Markdown 编辑器,支持所见即所得、即时渲染(类似 Typora)和分屏预览模式。它使用 TypeScript 实现,支持原生 JavaScript、Vue、React 和 Angular。

    352 引用 • 1815 回帖 • 3 关注
  • Pipe

    Pipe 是一款小而美的开源博客平台。Pipe 有着非常活跃的社区,可将文章作为帖子推送到社区,来自社区的回帖将作为博客评论进行联动(具体细节请浏览 B3log 构思 - 分布式社区网络)。

    这是一种全新的网络社区体验,让热爱记录和分享的你不再感到孤单!

    132 引用 • 1114 回帖 • 125 关注
  • Gitea

    Gitea 是一个开源社区驱动的轻量级代码托管解决方案,后端采用 Go 编写,采用 MIT 许可证。

    4 引用 • 16 回帖 • 4 关注
  • 黑曜石

    黑曜石是一款强大的知识库工具,支持本地 Markdown 文件编辑,支持双向链接和关系图。

    A second brain, for you, forever.

    15 引用 • 122 回帖
  • 酷鸟浏览器

    安全 · 稳定 · 快速
    为跨境从业人员提供专业的跨境浏览器

    3 引用 • 59 回帖 • 26 关注
  • LeetCode

    LeetCode(力扣)是一个全球极客挚爱的高质量技术成长平台,想要学习和提升专业能力从这里开始,充足技术干货等你来啃,轻松拿下 Dream Offer!

    209 引用 • 72 回帖 • 1 关注
  • 电影

    这是一个不能说的秘密。

    121 引用 • 601 回帖
  • jsoup

    jsoup 是一款 Java 的 HTML 解析器,可直接解析某个 URL 地址、HTML 文本内容。它提供了一套非常省力的 API,可通过 DOM,CSS 以及类似于 jQuery 的操作方法来取出和操作数据。

    6 引用 • 1 回帖 • 476 关注
  • GAE

    Google App Engine(GAE)是 Google 管理的数据中心中用于 WEB 应用程序的开发和托管的平台。2008 年 4 月 发布第一个测试版本。目前支持 Python、Java 和 Go 开发部署。全球已有数十万的开发者在其上开发了众多的应用。

    14 引用 • 42 回帖 • 764 关注
  • Sphinx

    Sphinx 是一个基于 SQL 的全文检索引擎,可以结合 MySQL、PostgreSQL 做全文搜索,它可以提供比数据库本身更专业的搜索功能,使得应用程序更容易实现专业化的全文检索。

    1 引用 • 214 关注
  • jsDelivr

    jsDelivr 是一个开源的 CDN 服务,可为 npm 包、GitHub 仓库提供免费、快速并且可靠的全球 CDN 加速服务。

    5 引用 • 31 回帖 • 59 关注
  • MySQL

    MySQL 是一个关系型数据库管理系统,由瑞典 MySQL AB 公司开发,目前属于 Oracle 公司。MySQL 是最流行的关系型数据库管理系统之一。

    691 引用 • 535 回帖
  • WebClipper

    Web Clipper 是一款浏览器剪藏扩展,它可以帮助你把网页内容剪藏到本地。

    3 引用 • 9 回帖 • 1 关注
  • Flume

    Flume 是一套分布式的、可靠的,可用于有效地收集、聚合和搬运大量日志数据的服务架构。

    9 引用 • 6 回帖 • 632 关注
  • 支付宝

    支付宝是全球领先的独立第三方支付平台,致力于为广大用户提供安全快速的电子支付/网上支付/安全支付/手机支付体验,及转账收款/水电煤缴费/信用卡还款/AA 收款等生活服务应用。

    29 引用 • 347 回帖 • 1 关注
  • Spark

    Spark 是 UC Berkeley AMP lab 所开源的类 Hadoop MapReduce 的通用并行框架。Spark 拥有 Hadoop MapReduce 所具有的优点;但不同于 MapReduce 的是 Job 中间输出结果可以保存在内存中,从而不再需要读写 HDFS,因此 Spark 能更好地适用于数据挖掘与机器学习等需要迭代的 MapReduce 的算法。

    74 引用 • 46 回帖 • 553 关注
  • Solidity

    Solidity 是一种智能合约高级语言,运行在 [以太坊] 虚拟机(EVM)之上。它的语法接近于 JavaScript,是一种面向对象的语言。

    3 引用 • 18 回帖 • 399 关注
  • BookxNote

    BookxNote 是一款全新的电子书学习工具,助力您的学习与思考,让您的大脑更高效的记忆。

    笔记整理交给我,一心只读圣贤书。

    1 引用 • 1 回帖 • 1 关注