设计模式之工厂模式VS抽象工厂
一、工厂模式主要是为创建对象提供过渡接口,以便将创建对象的具体过程屏蔽隔离起来,达到提高灵活性的目的。
工厂模式在《Java与模式》中分为三类:
1)简单工厂模式(Simple Factory):不利于产生系列产品;
2)工厂方法模式(Factory Method):又称为多形性工厂;
3)抽象工厂模式(Abstract Factory):又称为工具箱,产生产品族,但不利于产生新的产品;
这三种模式从上到下逐步抽象,并且更具一般性。
GOF在《设计模式》一书中将工厂模式分为两类:工厂方法模式(Factory Method)与抽象工厂模式(Abstract Factory)。将简单工厂模式(Simple Factory)看为工厂方法模式的一种特例,两者归为一类。
二、简单工厂模式
简单工厂模式又称静态工厂方法模式。重命名上就可以看出这个模式一定很简单。它存在的目的很简单:定义一个用于创建对象的接口。
1) 工厂类角色:这是本模式的核心,含有一定的商业逻辑和判断逻辑。在java中它往往由一个具体类实现。
2) 抽象产品角色:它一般是具体产品继承的父类或者实现的接口。在java中由接口或者抽象类来实现。
3) 具体产品角色:工厂类所创建的对象就是此角色的实例。在java中由一个具体类实现。
来看下它的组成:
1)抽象工厂角色: 这是工厂方法模式的核心,它与应用程序无关。是具体工厂角色必须实现的接口或者必须继承的父类。在java中它由抽象类或者接口来实现。
2)具体工厂角色:它含有和具体业务逻辑有关的代码。由应用程序调用以创建对应的具体产品的对象。
3)抽象产品角色:它是具体产品继承的父类或者是实现的接口。在java中一般有抽象类或者接口来实现。
4)具体产品角色:具体工厂角色所创建的对象就是此角色的实例。在java中由具体的类来实现。
工厂方法模式使用继承自抽象工厂角色的多个子类来代替简单工厂模式中的“上帝类”。正如上面所说,这样便分担了对象承受的压力;而且这样使得结构变得灵活 起来——当有新的产品(即暴发户的汽车)产生时,只要按照抽象产品角色、抽象工厂角色提供的合同来生成,那么就可以被客户使用,而不必去修改任何已有的代 码。可以看出工厂角色的结构也是符合开闭原则的!

- package cn.happy.b;
- //抽象产品角色
- public class LeiFeng {
- public void sweep(){
- System.out.println("扫地");
- }
- public void wash(){
- System.out.println("洗衣服");
- }
- }
- package cn.happy.b;
- //具体产品角色
- public class Student extends LeiFeng{
- }
- package cn.happy.b;
- //具体产品角色
- public class Volunter extends LeiFeng{
- }
- package cn.happy.b;
- //抽象工厂
- public interface IFactory {
- public LeiFeng getLeiFeng();
- }
- package cn.happy.b;
- //具体工厂
- public class StudentFactory implements IFactory{
- @Override
- public LeiFeng getLeiFeng() {
- return new Student();
- }
- }
- package cn.happy.b;
- //具体工厂
- public class VolunterFactory implements IFactory{
- @Override
- public LeiFeng getLeiFeng() {
- return new Volunter();
- }
- }
- package cn.happy.b;
- //测试类
- public class Test {
- public static void main(String[] args) {
- IFactory factory=new StudentFactory();
- //IFactory factory=new VolunterFactory();
- LeiFeng stu=factory.getLeiFeng();
- stu.wash();
- stu.sweep();
- }
- }
工厂方法定义:
定义一个用于创建对象的接口,让子类决定实例化哪一个类。工厂方法使一个类的实例化延迟到了子类
在简单工厂中,工厂类与分支耦合,所以我们就对工厂类下手,把工厂类抽象出一个接口,
这个接口只有一个方法,就是创建抽象产品的工厂方法,然后所有要生产具体的工厂,都实现这个接口,这样简单工厂的工厂类就变成了一个工厂抽象接口和多个具体生产对象的工厂,要增加功能,就不需要更改原来的工厂类,之需要增加此功能的类和相应的工厂类就可以了
工厂方法模式和简单工厂模式在定义上的不同是很明显的。工厂方法模式的核心是一个抽象工厂类,而不像简单工厂模式, 把核心放在一个实类上。工厂方法模式可以允许很多实的工厂类从抽象工厂类继承下来, 从而可以在实际上成为多个简单工厂模式的综合,从而推广了简单工厂模式。
反过来讲,简单工厂模式是由工厂方法模式退化而来。设想如果我们非常确定一个系统只需要一个实的工厂类, 那么就不妨把抽象工厂类合并到实的工厂类中去。而这样一来,我们就退化到简单工厂模式了。
定义:
提供一个创建一系列相关或相互依赖对象的接口,而无须指定他们具体的类
抽象工厂模式:
多个抽象产品类,每个抽象产品类可以派生出多个具体产品类。
一个抽象工厂类,可以派生出多个具体工厂类。
每个具体工厂类可以创建多个具体产品类的实例。
抽象工厂模式与工厂方法模式的区别
抽象工厂模式是工厂方法模式的升级版本,他用来创建一组相关或者相互依赖的对象。他与工厂方法模式的区别就在于,工厂方法模式针对的是一个产品等级结构;而抽象工厂模式则是针对的多个产品等级结构。在编程中,通常一个产品结构,表现为一个接口或者抽象类,也就是说,工厂方法模式提供的所有产品都是衍生自同一个接口或抽象类,而抽象工厂模式所提供的产品则是衍生自不同的接口或抽象类。
在抽象工厂模式中,有一个产品族的概念:所谓的产品族,是指位于不同产品等级结构中功能相关联的产品组成的家族。抽象工厂模式所提供的一系列产品就组成一个产品族;而工厂方法提供的一系列产品称为一个等级结构。我们依然拿生产汽车的例子来说明他们之间的区别。
在上面的类图中,User和Student是两个抽象产品,之所以为抽象,是因为他们都有可能有两种不同的实现,而SqlUser、AccessUser和SqlStudent、AccessStudent
就是对这两个抽象产品的具体分类
IFactory是一个抽象工厂接口,他里面应该包含所有产品创建的抽象方法,SqlFactory和AccessFatory就是具体工厂;
代码:
- package cn.happy.c;
- public class User {
- private int id;
- private String name;
- public int getId() {
- return id;
- }
- public void setId(int id) {
- this.id = id;
- }
- public String getName() {
- return name;
- }
- public void setName(String name) {
- this.name = name;
- }
- }
- package cn.happy.c;
- public class Student {
- private int id;
- private String name;
- public int getId() {
- return id;
- }
- public void setId(int id) {
- this.id = id;
- }
- public String getName() {
- return name;
- }
- public void setName(String name) {
- this.name = name;
- }
- }
- package cn.happy.c;
- /**
- * IUser接口,用于客户端访问,解除与具体数据库访问的耦合
- * @author 川哥哥
- *
- */
- public interface IUser {
- public void insert(User user);
- public User getUser(int id);
- }
- package cn.happy.c;
- public interface IStudent {
- public void insert(Student stu);
- public Student getUser(int id);
- }
- package cn.happy.c;
- public class SqlUser implements IUser{
- @Override
- public void insert(User user) {
- System.out.println("sql中给user表添加一条记录");
- }
- @Override
- public User getUser(int id) {
- System.out.println("在sql中根据id得到一条记录");
- return null;
- }
- }
- package cn.happy.c;
- public class AccessUser implements IUser{
- @Override
- public void insert(User user) {
- System.out.println("Access中给user表添加一条记录");
- }
- @Override
- public User getUser(int id) {
- System.out.println("在Access中根据id得到一条记录");
- return null;
- }
- }
- package cn.happy.c;
- public class SqlStudent implements IStudent{
- @Override
- public void insert(Student stu) {
- System.out.println("在sql中给student表添加一条记录");
- }
- @Override
- public Student getUser(int id) {
- System.out.println("在sql中,根据Id得到一条记录");
- return null;
- }
- }
- package cn.happy.c;
- public class AccessStudent implements IStudent{
- @Override
- public void insert(Student stu) {
- System.out.println("在Access中给student表添加一条记录");
- }
- @Override
- public Student getUser(int id) {
- System.out.println("在Access中,根据Id得到一条记录");
- return null;
- }
- }
- package cn.happy.c;
- /**
- * 定义一个接口,创建访问user表对象的抽象工厂类
- * @author 川哥哥
- *
- */
- public interface IFactory {
- public IUser creIUser();
- public IStudent greateStudent();
- }
- package cn.happy.c;
- /**
- * 实现Ifactory接口,实例化SqlUser
- * @author 川哥哥
- *
- */
- public class
- SqlFactory implements IFactory{
- @Override
- public IUser creIUser() {
- return new SqlUser();
- }
- @Override
- public IStudent greateStudent() {
- // TODO Auto-generated method stub
- return new SqlStudent();
- }
- }
- package cn.happy.c;
- public class AccessFactory implements IFactory{
- @Override
- public IUser creIUser() {
- return new AccessUser();
- }
- @Override
- public IStudent greateStudent() {
- // TODO Auto-generated method stub
- return new AccessStudent();
- }
- }
测试类
- package cn.happy.c;
- public class Test {
- public static void main(String[] args) {
- User user=new User();
- Student stu=new Student();
- IFactory factory=new SqlFactory();
- //IFactory factory=new AccessFactory();
- IUser iUser=factory.creIUser();
- IStudent is=factory.greateStudent();
- iUser.insert(user);
- iUser.getUser(1);
- is.insert(stu);
- is.getUser(1);
- }
- }
运行结果:
sql中给user表添加一条记录
在sql中根据id得到一条记录
在sql中给student表添加一条记录
在sql中,根据Id得到一条记录
抽象工厂模式的优点
抽象工厂模式除了具有工厂方法模式的优点外,最主要的优点就是可以在类的内部对产品族进行约束。所谓的产品族,一般或多或少的都存在一定的关联,抽象工厂模式就可以在类内部对产品族的关联关系进行定义和描述,而不必专门引入一个新的类来进行管理。
抽象工厂模式的缺点
产品族的扩展将是一件十分费力的事情,假如产品族中需要增加一个新的产品,则几乎所有的工厂类都需要进行修改。所以使用抽象工厂模式时,对产品等级结构的划分是非常重要的。
适用场景
当需要创建的对象是一系列相互关联或相互依赖的产品族时,便可以使用抽象工厂模式。说的更明白一点,就是一个继承体系中,如果存在着多个等级结构(即存在着多个抽象类),并且分属各个等级结构中的实现类之间存在着一定的关联或者约束,就可以使用抽象工厂模式。假如各个等级结构中的实现类之间不存在关联或约束,则使用多个独立的工厂来对产品进行创建,则更合适一点。
总结
无论是简单工厂模式,工厂方法模式,还是抽象工厂模式,他们都属于工厂模式,在形式和特点上也是极为相似的,他们的最终目的都是为了解耦。在使用时,我们不必去在意这个模式到底工厂方法模式还是抽象工厂模式,因为他们之间的演变常常是令人琢磨不透的。经常你会发现,明明使用的工厂方法模式,当新需求来临,稍加修改,加入了一个新方法后,由于类中的产品构成了不同等级结构中的产品族,它就变成抽象工厂模式了;而对于抽象工厂模式,当减少一个方法使的提供的产品不再构成产品族之后,它就演变成了工厂方法模式。
所以,在使用工厂模式时,只需要关心降低耦合度的目的是否达到了。
设计模式之工厂模式VS抽象工厂的更多相关文章
- Delphi 设计模式:《HeadFirst设计模式》Delphi代码---工厂模式之抽象工厂[转]
1 2 {<HeadFirst设计模式>工厂模式之抽象工厂 } 3 { 抽象工厂的产品 } 4 { 编译工具:Delphi7.0 ...
- C# 设计模式(1)——简单工厂模式、工厂模式、抽象工厂模式
1.前言 上一篇写了设计模式原则有助于我们开发程序的时候能写出高质量的代码(牵一发而不动全身),这个系列还是做个笔记温习一下各种设计模式,下面就看看简单工厂模式.工厂模式.抽象工厂模式. 2.简单工厂 ...
- C#设计模式--工厂模式和抽象工厂模式
话说有三大潮牌公司一直相互PK,有一天举办了一个活动让这三大公司来一个PK,我们来看看哪家公司的上衣做出来好看穿得舒服 现在我们有一个上衣的抽象产品让三大公司来做 //抽象产品 public inte ...
- JAVA设计模式 3【创建型】理解工厂模式与抽象工厂模式
上一节我们已经学习了原型模式,稍微复习一下:通过重写Object 类的clone() 方法实现浅克隆,浅克隆也要实现Cloneable 标记接口.而深克隆则是将对象通过序列化和反序列化 的方式进行创建 ...
- c#工厂模式与抽象工厂模式
一. 工厂方法(Factory Method)模式 工厂方法(FactoryMethod)模式是类的创建模式,其用意是定义一个创建产品对象的工厂接口,将实际创建工作推迟到子类中. 工厂方法模式是简单工 ...
- Objective-C 工厂模式(下) -- 抽象工厂模式
相比简单工厂模式, 只有一个工厂 能生产的手机也是固定的 抽象工厂模式类似于有很多家工厂, 当用户要买什么手机就创建对应的工厂去生产 比如用户要买iPhone就创建一个Apple工厂来生产手机, 要买 ...
- 工厂模式[3] 抽象工厂 Abstract Factory
简介 1.简单工厂,或静态工厂,产品接口 定义:专门定义一个类来负责创建其他类的实例,被创建的实例通常具有共同的父类或实现同一接口 优点:客户端可以直接消费产品,而不必关心具体产品的实现(不关心对象的 ...
- [19/04/23-星期二] GOF23_创建型模式(工厂模式、抽象工厂模式)
一.工厂模式(分为:简单工厂模式.工厂方法模式.抽象工厂模式) 实现了创建者和调用者的分离 核心本质:1.实例化对象,用工厂方法代替new操作:2.将选择实现类.创建对象统一管理和控制,从而将调用者跟 ...
- factory工厂模式之抽象工厂AbstractFactory
* 抽象工厂: 意图在于创建一系列互相关联或互相依赖的对象. * 每个工厂都会创建一个或多个一系列的产品 * 适用于:产品不会变动,开始时所有产品都创建好,然后根据分类获取想要的 某一类产品(很像sp ...
随机推荐
- 《Web 前端面试指南》1、JavaScript 闭包深入浅出
闭包是什么? 闭包是内部函数可以访问外部函数的变量.它可以访问三个作用域:首先可以访问自己的作用域(也就是定义在大括号内的变量),它也能访问外部函数的变量,和它能访问全局变量. 内部函数不仅可以访问外 ...
- 微信小程序开发心得
微信小程序也已出来有一段时间了,最近写了几款微信小程序项目,今天来说说感受. 首先开发一款微信小程序,最主要的就是针对于公司来运营的,因为,在申请appid(微信小程序ID号)时候,需要填写相关的公司 ...
- 移动站应该尝试百度MIP的五个原因
MIP是什么?MIP是百度在2016年提出的移动网页加速器项目. MIP能做什么?MIP能帮助站长和网站开发者快速搭建移动端页面. MIP怎么加速?MIP从前端渲染和页面网络传输两方面进行优化,杜绝页 ...
- Android-armebi-v7a、arm64-v8a、armebi的坑
先来一波扫盲: armeabi:针对普通的或旧的arm v5 cpu armeabi-v7a:针对有浮点运算或高级扩展功能的arm v7 cpu(32位ARM设备) arm64-v8a:64位ARM设 ...
- 卡片抽奖插件 CardShow
这个小项目(卡片秀)是一个卡片抽奖特效插件,用开源项目这样的词语让我多少有些羞愧,毕竟作为一个涉世未深的小伙子,用项目的标准衡量还有很大差距.不过该案例采用 jQuery 插件方式编写,提供配置参数并 ...
- 从Vue.js窥探前端行业
近年来前端开发趋势 1.旧浏览器逐渐淘汰,移动端需求增加: 旧浏览器主要指的是IE6-IE8,它是不支持ES5特性的:IE9+.chrome.sarafi.firefox对ES5是完全支持的,移动端大 ...
- [原] KVM虚拟机网络闪断分析
背景 公司云平台的机器时常会发生网络闪断,通常在10s-100s之间. 异常情况 VM出现问题时,表现出来的情况是外部监控系统无法访问,猜测可能是由于系统假死,OVS链路问题等等.但是在出现网络问题的 ...
- LINQ to SQL Select查询
1. 查询所有字段 using (NorthwindEntities context = new NorthwindEntities()) { var order = from n in contex ...
- 【WPF】日常笔记
本文专用于记录WPF开发中的小细节,作为备忘录使用. 1. 关于绑定: Text ="{Binding AnchorageValue,Mode=TwoWay,UpdateSourceTrig ...
- python中IndentationError: expected an indented block错误的解决方法
IndentationError: expected an indented block 翻译为IndentationError:预期的缩进块 解决方法:有冒号的下一行要缩进,该缩进就缩进