提供一个创建一系列或相互依赖对象的接口,而无须指定他们具体的类。例如某些系统可能需要为用户提供一系列相关对象,但系统不希望用户直接使用new运算符实例化这些对象,而是应当由系统来控制这些对象的创建,否则用户不仅要清楚知道哪些类来创建对象,而且必须要清楚这些对象之间是如何相关的,使得用户代码和这些类型形成耦合,不利维护,一般会包括以下四种角色:

  • 抽象产品(Product):一个抽象类或接口,负责定义具体产品必须实现的方法;
  • 具体产品(ConcreteProduct):具体产品是一个类,如果Product是一个抽象类,那么具体产品是它的子类;如果Product是一个接口,那么具体产品是它的接口类;
  • 抽象工厂(AbstractFactory):一个接口或抽象类,负责定义若干个抽象方法;
  • 具体工厂(ConcreteFactory):如果抽象工厂是一个抽象类,具体工厂就是它的子类;如果是个接口就是它的实现类。具体工厂重写抽象工厂中的抽象方法,使该方法返回具体的产品实例。

  抽象工厂的UML类图如下所示:

  抽象工厂模式的优点:

    • 可以为用户创建一系列相关的对象,使用户和创建这些对象的类脱藕。
    • 可以方便的为用户配置一系列对象,用户使用不同的具体工厂就能得到一组相关的对象,同时也能避免用户混用不同系列中的对象。
    • 可以随时增加具体工厂为用户提供一组相关的对象。

  抽象工厂模式的应用场景:

    • 系统需要为用户提供多个对象,但不希望用户直接使用new对象,希望用户和创建对象类的解耦
    • 系统需要为用户提供多个相关的对象,以便用户联合使用他们,但又不希望用户来决定这些对象是如何关联的。
    • 系统需要为用户提供一系列对象,但值需要用户知道这些对象有哪些方法可以用,不需要用户知道这些对象的创建过程

下面以服饰工厂生产衣服为例,为用户提供西装和牛仔裤:

1. 抽象产品,裤子和上衣

public abstract class Trousers {

    public abstract int getWaitstSize();
public abstract int getHeight();
public abstract String getName();
}
public abstract class UpperClothes {

    public abstract int getChestSize();
public abstract int getHeight();
public abstract String getName();
}

2. 具体产品

public class CowBoyTrouers  extends Trousers {
private int waistSize;
private int height;
private String name; public CowBoyTrouers(int waistSize, int height, String name) {
this.waistSize = waistSize;
this.height = height;
this.name = name;
} @Override
public int getWaitstSize() {
return waistSize;
} @Override
public int getHeight() {
return height;
} @Override
public String getName() {
return name;
} }
public class WesternTrouers extends Trousers {

    private int waistSize;
private int height;
private String name; public WesternTrouers(int waistSize, int height, String name) {
this.waistSize = waistSize;
this.height = height;
this.name = name;
} @Override
public int getWaitstSize() {
return waistSize;
} @Override
public int getHeight() {
return height;
} @Override
public String getName() {
return name;
}
}
public class CowBoyUpperClothes extends UpperClothes {

    private int chextSize;
private int height;
private String name; public CowBoyUpperClothes(int chextSize, int height, String name) {
super();
this.chextSize = chextSize;
this.height = height;
this.name = name;
} @Override
public int getHeight() {
return height;
} @Override
public String getName() {
return name;
} @Override
public int getChestSize() {
return chextSize;
}
}
public class WesternUpperClothes extends UpperClothes {

    private int chestSize;
private int height;
private String name; public WesternUpperClothes(int chestSize, int height, String name) {
super();
this.chestSize = chestSize;
this.height = height;
this.name = name;
} @Override
public int getChestSize() {
return chestSize;
} @Override
public int getHeight() {
return height;
} @Override
public String getName() {
return name;
} }

3. 抽象工厂

public abstract class ClotherFactory {

    public abstract UpperClothes createUpperClothes(int chestSize,int height);
public abstract Trousers createTrousers(int waistSize,int height);
}

4.具体工厂,一个负责生产西装,一个负责制作牛仔

public class BJClotherFactory extends ClotherFactory {

    @Override
public UpperClothes createUpperClothes(int chestSize, int height) {
return new WesternUpperClothes(chestSize,height,"北京牌西服上衣");
} @Override
public Trousers createTrousers(int waistSize, int height) {
return new WesternTrouers(waistSize, height,"北京牌西服裤子");
} }
public  class SHClothesFactory extends ClotherFactory {

    @Override
public UpperClothes createUpperClothes(int chestSize, int height) {
return new CowBoyUpperClothes(chestSize, height, "上海牌牛仔上衣");
} @Override
public Trousers createTrousers(int waistSize, int height) {
return new CowBoyTrouers(waistSize, height, "上海牌牛仔裤子");
} }

上面这些类就是一个小框架,可以使用这个小框架编写自己的类。应用程序在使用抽象工厂模式时,只和抽象的产品、抽象工厂以及具体工厂打交道,用户只需要了解抽象产品有哪些方法即可,不需要知道有哪些具体产品。下列代码列出了一个应用程序的类。

public class Shop {

    private UpperClothes cloth;
private Trousers trouser;
public void giveSuit(ClotherFactory factory,int chestSize,int waistSize,int height){
this.cloth = factory.createUpperClothes(chestSize, height);
this.trouser = factory.createTrousers(waistSize, height);
showMess();
}
private void showMess() {
System.out.println("<套装信息>");
System.out.println(cloth.getName()+":");
System.out.print("胸围:"+cloth.getChestSize());
System.out.println(" 身高:"+cloth.getHeight());
System.out.println(trouser.getName()+":");
System.out.print("腰围:"+trouser.getWaitstSize());
System.out.println(" 身高"+trouser.getHeight());
}
}
public class Application {

    public static void main(String[] args) {
Shop shop = new Shop();
ClotherFactory facory = new BJClotherFactory();
shop.giveSuit(facory, 110, 82, 170);
facory = new SHClothesFactory();
shop.giveSuit(facory, 120, 88, 180);
}
}

Java的设计模式(4)--抽象工厂模式的更多相关文章

  1. Java 设计模式之抽象工厂模式(三)

    原文地址:Java 设计模式之抽象工厂模式(三) 博客地址:http://www.extlight.com 一.前言 上篇文章 <Java 设计模式之工厂模式(二)>,介绍了简单工厂模式和 ...

  2. 乐在其中设计模式(C#) - 抽象工厂模式(Abstract Factory Pattern)

    原文:乐在其中设计模式(C#) - 抽象工厂模式(Abstract Factory Pattern) [索引页][源码下载] 乐在其中设计模式(C#) - 抽象工厂模式(Abstract Factor ...

  3. 桥接模式及C++实现 C++设计模式-AbstractFactory抽象工厂模式

    桥接模式及C++实现 桥接模式 先说说桥接模式的定义:将抽象化(Abstraction)与实现化(Implementation)分离,使得二者可以独立地变化. 桥接模式号称设计模式中最难理解的模式之一 ...

  4. java设计模式之抽象工厂模式

    上一篇文章(http://www.cnblogs.com/liaoweipeng/p/5768197.html)讲了简单工厂模式,但是简单工厂模式存在一定的问题,如果想要拓展程序,必须对工厂类进行修改 ...

  5. Java设计模式系列-抽象工厂模式

    原创文章,转载请标注出处:https://www.cnblogs.com/V1haoge/p/10755412.html 一.概述 抽象工厂模式是对工厂方法模式的再升级,但是二者面对的场景稍显差别. ...

  6. java设计模式(三)--抽象工厂模式

    转载:http://zz563143188.iteye.com/blog/1847029 前面的工厂方法模式虽然清晰,但还是感觉有些繁琐,通常使用的还是抽象工厂模式. 工厂方法模式有一个问题就是,类的 ...

  7. Java 设计模式之抽象工厂模式

    抽象工厂模式(Abstract Factory Pattern)是围绕一个超级工厂创建其他工厂.该超级工厂又称为其他工厂的工厂.这种类型的设计模式属于创建型模式,它提供了一种创建对象的最佳方式. 在抽 ...

  8. [java] java 设计模式(2):抽象工厂模式(Abstract Factory)

    工厂方法模式有一个问题就是,类的创建依赖工厂类,也就是说,如果想要拓展程序,必须对工厂类进行修改,这违背了闭包原则,所以,从设计角度考虑,有一定的问题,如何解决?就用到抽象工厂模式,创建多个工厂类,这 ...

  9. 设计模式之抽象工厂模式(Java实现)

    “上次是我的不对,贿赂作者让我先讲来着,不过老婆大人大人有大量,不与我计较,这次还让我先把上次未讲完的应用场景部分给补充上去,有妻如此,夫复何求.”(说完,摸了摸跪的发疼的膝盖,咳咳,我发四我没笑!真 ...

  10. Java设计模式(3)——抽象工厂模式

    抽象工厂模式是所有形态的工厂模式中最为抽象和最其一般性的.抽象工厂模式可以向客户端提供一个接口,使得客户端在不必指定产品的具体类型的情况下,能够创建多个产品族的产品对象. 一.产品族和产品等级结构 为 ...

随机推荐

  1. Java的 「 “ 结构体 ”」 与 「 “ 自定义排序 ” 」

    Java里面的结构体可以靠class来实现,如果相对结构体进行排序,需要写一个接口,class 自定义的名字 implements Comparator<结构体(自己定义的class类的名字)& ...

  2. (4)打造简单OS-loader硬盘加载和C++写入文件

    0.简要说明: 我们完全可以使用bochs创建映像文件,如https://blog.csdn.net/jadeshu/article/details/89046838   ,那么为什么还去用C++去模 ...

  3. PHP 连接本地mysql

    <?php echo microtime(true); ?> <?php $servername = "localhost"; $username = " ...

  4. Python的十种常见算法

    十种排序算法 1. 常见算法分类 十种常见排序算法一般分为以下几种: (1)非线性时间比较类排序: ​ a. 交换类排序(快速排序.冒泡排序) ​ b. 插入类排序(简单插入排序.希尔排序) ​ c. ...

  5. (转)SLOW READPROCESSOR;ERROR SLOW BLOCKRECEIVER错误日志分析

    1.总结 "Slow ReadProcessor" 和"Slow BlockReceiver"往往是因为集群负载比较高或者某些节点不健康导致的,本文主要是帮助你 ...

  6. WebSocketSharp 创建客户端和服务端

    这里没有对onOpen.onClose.onError做案例,生产环境需要具备. 1.客户端 只推送不接收数据 创建WebSocketClient类 class WebSocketClient { W ...

  7. 总结Lock和synchronized的区别

    1. Lock是一个接口,而synchronized是Java中的关键字,synchronized是内置的语言实现,Lock是代码层面的实现. 2. Lock可以选择性的获取锁,如果一段时间获取不到, ...

  8. Qwidget::update

    void QWidget::update ()分析重绘事件激活 1看看手册中这段话 void QWidget::update () [slot] Updates the widget unless u ...

  9. jmeter 随机取一个值的方法

    1.添加用户自定义变量 在要用到随机值的地方写入 ${__RandomFromMultipleVars(1|2|0)} 例子: 效果:

  10. ABAP Memory ID

    转自:https://blog.csdn.net/lyq123333321/article/details/52659114 (一)          Difference Between SAP a ...