【设计模式】工厂模式
创始人
2024-05-30 20:31:16
0

工厂模式

所谓工厂,顾名思义,就是创建出一类相似的产品的,工厂模式可以帮我们创建各个复杂/简单对象。属于创建型模式。

工厂模式分为三类:

  • 简单工厂
  • 工厂方法
  • 抽象工厂

简单工厂

比方说我们需要根据配置文件去解析配置,不同后缀名的文件需要使用不同的解析器去解析,我们当然可以将所有的代码写在一个方法里
也可以使用简单工厂,将根据文件后缀名选择解析器的逻辑封装在工厂类中

定义一个IConfigParser接口,然后定义三个不同的解析器
在这里插入图片描述

public interface IConfigParser {Config parse(String configText);
}
public class JsonConfigParser implements IConfigParser{@Overridepublic Config parse(String configText) {return new Config("json配置文件");}
}
public class PropertiesConfigParser implements IConfigParser{@Overridepublic Config parse(String configText) {return new Config("properties配置文件");}
}
public class XmlConfigParser implements IConfigParser{@Overridepublic Config parse(String configText) {return new Config("xml配置文件");}
}
public class ConfigParserFactory {public static IConfigParser createParser(String configFormat) {if ("json".equalsIgnoreCase(configFormat)) {return new JsonConfigParser();} else  if ("xml".equalsIgnoreCase(configFormat)) {return new XmlConfigParser();} else  if ("properties".equalsIgnoreCase(configFormat)) {return new PropertiesConfigParser();}return null;}
}
public class Config {private String name;public Config(String name) {this.name = name;}
}

工厂方法模式

简单工厂适合于创建逻辑比较的场景,如果每个解析器的构建过程都十分复杂,那是不是有一个单独的工厂类去创建各自的解析器会更合适。为了去除上面if … else …的分支判断,我们将工厂对象提前缓存好,其类图如下:
在这里插入图片描述

将上述代码重构后:

public interface IConfigParserFactory {IConfigParser createParse();
}
public class JsonConfigParserFactory implements IConfigParserFactory{@Overridepublic IConfigParser createParse() {return new JsonConfigParser();}
}
public class XmlConfigParserFactory implements IConfigParserFactory{@Overridepublic IConfigParser createParse() {return new XmlConfigParser();}
}
public class Test {private static final Map factoryMap = new HashMap<>();static {factoryMap.put("json", new JsonConfigParserFactory());factoryMap.put("xml", new XmlConfigParserFactory());}public static void main(String[] args) {IConfigParser configParser = factoryMap.get("json").createParse();Config config = configParser.parse("{}");}
}

抽象工厂

抽象工厂模式与工厂方法模式的最大区别就在于,工厂方法模式针对的是一个产品等级结构;而抽象工厂模式则需要面对多个产品等级结构。

在学习抽象工厂具体实例之前,应该明白两个重要的概念:产品族和产品等级
所谓产品族,是指位于不同产品等级结构中,功能相关联的产品组成的家族。

以现实生活中的产品为例,我们有洗衣机,电风扇等电器,每种电器有各自的厂商,比方说美的的洗衣机、风扇,小米的洗衣机、风扇。我们认为一个品牌生产的东西是一个产品族的(美的的洗衣机和风扇都属于美的产品族),不同工厂生产的一类产品认为是一个产品等级结构的。

而所谓的抽象工厂,就是提供一个创建一系列相关或者相互依赖的对象的接口。在抽象工厂中,每一个产品族都有一个具体工厂,而每一个具体工厂负责创建属于同一个产品族,但是属于不同产品等级结构的产品。

回到前面配置解析的例子,不同的文件格式的解析器不一样,但是相同的配置文件(eg,json配置文件)在不同的系统和不同的规则下的解析器也不一样。如果按照工厂方法模式的方式来实现,给每个文件格式和系统,规则都创建Factory接口,那么我们就需要创建N*N个工厂类了,这样代码就太臃肿了。

我们可以让一个工厂来帮忙创建不同的对象(Json工厂可以帮我们创建对应的system解析器和规则解析器)

实现如下:

public interface IConfigParserFactory {IRuleConfigParser createRuleParse();ISystemConfigParser createSystemParse();}
public interface IRuleConfigParser {Config parse(String configText);
}
public interface ISystemConfigParser {Config parse(String configText);
}
public class JsonConfigParserFactory implements IConfigParserFactory {@Overridepublic IRuleConfigParser createFileParse() {return new JsonRuleConfigParser();}@Overridepublic ISystemConfigParser createSystemParse() {return new JsonSystemConfigParser();}
}
public class XmlConfigParserFactory implements IConfigParserFactory {@Overridepublic IRuleConfigParser createFileParse() {return new XmlRuleConfigParser();}@Overridepublic ISystemConfigParser createSystemParse() {return new XmlSystemConfigParser();}
}

总结:

简单工厂 :专门定义一个类(Factory)来负责创建其它类的实例,被创建的实例通常都具有共同的父类或实现同一个接口。
​工厂(方法)模式:定义一个用于创建对象的接口(工厂接口IConfigParserFactory),让子类或实现去决定实例化哪个类。
抽象工厂模式 :提供接口给客户端用以创建一系列相关或者相互依赖的对象(往往会有一个可以创建多个产品等级结构的产品的工厂)

使用场景:

  1. 如果代码中存在大量 if-else 分⽀判断,需要动态地根据不同的类型创建不同的对象。这种情形我们就可以考虑使用工厂模式,将创建的逻辑抽离出来,放到工厂类中

  2. 不需要根据不同类型创建不同的对象,但是如果对象的创建过程十分复杂,那么我们也可以将其 创建过程封装到工厂类中

如果创建逻辑比较简单,那么直接使用简单工厂就可以了;如果创建逻辑比较复杂,那么就建议使用工厂方法模式

工厂模式的作⽤⽆外乎下⾯四个。这也是判断要不要使⽤⼯⼚模式的最本质的参考标准。

  1. 封装变化:创建逻辑有可能变化,封装成⼯⼚类之后,创建逻辑的变更对调⽤者透明。
  2. 代码复⽤:创建代码抽离到独⽴的⼯⼚类之后可以复⽤。
  3. 隔离复杂性:封装复杂的创建逻辑,调⽤者⽆需了解如何创建对象。
  4. 控制复杂度:将创建代码抽离出来,让原本的函数或类职责更单⼀,代码更简洁。

实际使用的时候不必太在意使用的究竟是简单工厂还是工厂方法或者抽象工厂,只要通过工厂模式可以使我们的代码易于扩展和维护就好了,没必要过度设计。

相关内容

热门资讯

巴基斯坦总理与巴各政党领导人通... 【环球网快讯】印巴局势升级引发广泛关注。据巴基斯坦《黎明报》10日援引巴基斯坦国家电视台PTV最新消...
这一天!邯郸市中心医院…… 转自:邯郸新闻网邯郸道 天使情——邯郸市中心医院的特色护士节活动编者按:在国际护士节来临之际,邯郸市...
泰国4月份通胀率现负增长 转自:中国新闻网  中新社曼谷5月10日电 据泰国《民族报》10日报道,泰国4月份通货膨胀率为-0....
5月12日现场对接!辽宁省工信... 原标题:需求清单,向全球发布!  需求是市场,是合作的开始。近日,省工业信息化厅围绕重点产业链条、工...
特朗普高兴早了!首位美籍罗马教... 当地时间5月8日,枢机主教罗伯特·普雷沃斯特当选第267任天主教罗马教皇,称为利奥十四世,成为首位美...
马光远谈中国文化IP:中国每一...   2025世界IP经济发展大会暨全球IP授权博览会将于5月10日-5月12日在广州举行。著名经济学...
媒体评:学术近亲繁殖伤害教育公... 【媒体评:#学术近亲繁殖伤害教育公平#】近日,因发表14篇SCI论文,并获得3项国家发明专利,重庆大...
熟练到脑子跟不上嘴!护理专业男... 【熟练到脑子跟不上嘴!#护理专业男生丝滑操作婴儿急救#[666]】5月9日,四川一护理专业男同学,在...
深化改革开放再发力 国务院部署... 深化对外开放,国务院又有新部署。5月9日,国务院常务会议听取推动自贸试验区建设提质增效工作汇报,并研...
46岁章子怡,突传喜讯! 今天,章子怡工作室发文官宣好消息:《卧虎藏龙》25周年之际,章子怡在GoldGala出席奥斯卡博物馆...
面板双虎4月营收持平,Q2电视... 来源:Wit Display面板双虎友达光电今天公布4月合并营收新台币231.4亿元新台币,较3月减...
16楼“飞”出电竞椅!原因查明... 据四川成都市公安局高新技术产业开发区分局消息近日成都高新区某小区发生了一起高空抛物事件一男子情绪失控...
富采Q1亏1亿! 来源:Wit DisplayLED大厂富采5月9日举行法说会,公布2025年第1季财报。尽管单季仍呈...
小市值、反转因子表现较好   炒股就看金麒麟分析师研报,权威,专业,及时,全面,助您挖掘潜力主题机会! 因子IC跟踪    ...
全球专家共话科技与产业融合的脑... 5月10日,“2025浦江创新论坛”第七届神经科技国际创新论坛暨中国神经科学学会神经调控基础与转化分...
璀璨夺目!这场焰火展演点亮南昌... 转自:南昌发布5月9日晚,随着暮色降临,“星焰焕新·城启璀璨”——中国南昌烟花爆竹产业博览会城市焰火...
5月楼市开局冷热分化 公积金利... (转自:地产红榜)  5月1日国家新版《住宅项目规范》正式实施。从“五一”假期楼市表现来看,分化明显...
清华大学夏清:新型电力系统的构...   5月10日消息,远东控股集团40周年庆暨企业家论坛今天在宜兴举行。清华大学电机系教授、国家能源互...
十二载从无到有 海军舰载航空兵... 今天(5月10日)是人民海军首支舰载航空兵部队成立12周年的日子。12年前,舰载航空兵部队在渤海湾畔...
今天,该如何看尖岗山? 来源:苗头有朋友打来电话,说完全同意我的意见,全款买了一套万科未来之光80平方米的房子。有时候,你不...