设计模式二十四章经之工厂模式

概述

工厂模式在我们平常开发中应用比较广泛,或许你不知道,但你已经使用了很多次了。我们明确地计划不同条件下创建不同实例时,就会使用到工厂模式。工厂模式分为简单工厂模式和抽象工厂模式。这边我们先说简单的工厂模式。

使用场景

在任何需要生成复杂的对象的地方,都可以使用工厂模式。复杂对象适合使用工厂模式。而直接通过new就可以完成创建的对象无需使用工厂模式。

具体实现

首先,我们创建一个实体类的接口:

public interface Shape {
   void draw();
}

现在我们创建2个实体类去实现这个接口:

public class Square implements Shape {
   @Override
   public void draw() {
      System.out.println("Square:draw() method.");
   }
}
public class Circle implements Shape {
   @Override
   public void draw() {
      System.out.println("Circle:draw() method.");
   }
}

现在我们创建一个抽象工厂类:

public abstract class Factroy{
      public abstract Shape create();
}

然后我们创建工厂1和工厂2:

public class SquareFactory extends Factroy{
  @override
  public Shape create(){
      return new Square();
  }
}
public class CircleFactory extends Factroy{
  @override
  public Shape create(){
      return new Circle();
  }
}

现在我们就可以调用了:

public clas Demo{
  public static void main(String[] args){
    Factory factory1=new SquareFactory();
    Square square=factory1.create();
    square.draw();
    Factory factory2=new CircleFactory();
    Circle circle=factory2.create();
    circle.draw();
  }
}

我们可以看到输出结果如下:

Square:draw() method.
Circle:draw() method.

这边我们分成四个模块。一个是抽象产品,是所有产品的父类。第二是具体产品,我们需要通过实现抽象产品来实现具体产品的对象。第三是抽象工厂,其方法是工厂方法的核心。最后是具体工厂,实现抽象工厂去实现具体的逻辑。

不过我们发现具体工厂越来越多的时候我们的类会越来越多….那么我们可以通过反射去获取实例。仅需一个具体工厂类就可以解决了。那么我们把代码改一下:

public abstract class Factroy{
    public abstract<T extends Shape> T create(Class<T> clz);
}
public class BaseFactory extends Factroy{
  @override
    public <T extends Shape> T create(Class<T> clz){
      Shape shape=null;
      try{
          shape=(Shape)Class.forName(clz.getName()).newInstance();
      } catch(Exception e){
          e.printStackTrace();
      }
      return (T)shape;
  }
}

那么我们的main方法就可以这么修改了:

public clas Demo{
  public static void main(String[] args){
    Factory factory=new BaseFactory();
    Shape square=factory.create(Square.class);
    square.draw();
    Shape circle=factory.create(Circle.class);
    circle.draw();
  }
}

输出结果为:

Square:draw() method.
Circle:draw() method.

总结

工厂模式是完全符合设计原则的。降低了对象间的耦合度。而且他依赖于抽象的架构,将实例化的任务交给子类去完成。有较好的扩展性。

优点:

  • 一个调用者想创建一个对象,只要知道其名称就可以了。
  • 扩展性高,如果想增加一个产品,只要扩展一个工厂类就可以。
  • 屏蔽产品的具体实现,调用者只关心产品的接口。

缺点:

  • 每次增加一个产品时,都需要增加一个具体类和对象实现工厂,使得系统中类的个数成倍增加,在一定程度上增加了系统的复杂度,同时也增加了系统具体类的依赖。这并不是什么好事。

原文发布于微信公众号 - 我就是马云飞(coding_ma)

原文发表时间:2018-05-14

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏racaljk

C++11新语法糖之尾置返回类型

C++11的尾置返回类型初衷是为了方便复杂函数的声明和定义,但是当复杂度稍微提升一些的时候很明显能注意到这种设计的作用微乎其微.

1665
来自专栏诸葛青云的专栏

一步一步教你从零开始写C语言链表——构建一个链表

今天小编给大家带来c语言难点--链表的讲解,一步一步教你从零开始写C语言链表---构建一个链表。

1360
来自专栏Python小屋

Python标准库collections中与字典有关的类

Python标准库中提供了很多扩展功能,大幅度提高了开发效率。这里主要介绍OrderedDict类、defaultdict类和Counter类。 (1)Orde...

2896
来自专栏C语言及其他语言

C语言的数据类型

变量与常量数据 在程序的世界中,可以让计算机按照指令做很多事情,如进行数值计算、图像显示、语音对话、视频播放、天文计算、发送邮件、游戏绘图以及任何我们可以想...

5145
来自专栏架构师小秘圈

深入理解Java的接口和抽象类

架构技术干货第一时间送达! 对于面向对象编程来说,抽象是它的一大特征之一。在Java中,可以通过两种形式来体现OOP的抽象:接口和抽象类。这两者有太多相似的地方...

3405
来自专栏Java学习网

Java中有关Null的9问题

Java中有关Null的9问题 对于Java程序员来说,null是令人头痛的东西。时常会受到空指针异常(NPE)的骚扰。连Java的发明者都承认...

2305
来自专栏web前端教室

javascript 红皮高程(10)

继续string类型的相关知识哈,不细看不知道啊,这JS的知识点真是太细碎了。因为许多知识点都互相交织着,但某些属性却并不是所有的对象都有。例如: 转换字符...

2027
来自专栏用户画像

7.5.2 基数排序

基数排序是一种很特别的排序方法,它不是基于比较进行排序的,而是采用多关键字排序思想,借助“分配”和“收集”两种操作对单逻辑关键字进行排序。基数排序又分为最高位优...

803
来自专栏老马说编程

(19) 接口的本质 / 计算机程序的思维逻辑

数据类型的局限 之前我们一直在说,程序主要就是数据以及对数据的操作,而为了方便操作数据,高级语言引入了数据类型的概念,Java定义了八种基本数据类型,而类相当...

21010
来自专栏开源优测

Python3希尔排序

希尔排序 概述 希尔排序(Shell's Sort)是插入排序的一种又称“缩小增量排序”(Diminshing Increment Sort),是直接插入排序算...

35510

扫码关注云+社区

领取腾讯云代金券