提供一个创建一系列或相互依赖对象的接口,而无须指定他们具体的类。例如某些系统可能需要为用户提供一系列相关对象,但系统不希望用户直接使用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. SpringBoot JPA懒加载异常 - com.fasterxml.jackson.databind.JsonMappingException: could not initialize proxy

    问题与分析 某日忽然发现在用postman测试数据时报错如下: com.fasterxml.jackson.databind.JsonMappingException: could not initi ...

  2. Java Heap dump文件分析工具jhat简介

    jhat 是Java堆分析工具(Java heap Analyzes Tool). 在JDK6u7之后成为标配. 使用该命令需要有一定的Java开发经验,官方不对此工具提供技术支持和客户服务. 用法: ...

  3. ICEM-带死角弯管

    原视频下载地址:https://yunpan.cn/cqRiHaQiLi8I7  访问密码 b5c6

  4. Mininet系列实验(四):基于Mininet测量路径的损耗率

    1 实验目的 熟悉Mininet自定义拓扑脚本的编写与损耗率的设定: 熟悉编写POX脚本,测量路径损耗速率 2 实验原理 在SDN环境中,控制器可以通过对交换机下发流表操作来控制交换机的转发行为,此外 ...

  5. 【分类算法】决策树(Decision Tree)

    (注:本篇博文是对<统计学习方法>中决策树一章的归纳总结,下列的一些文字和图例均引自此书~) 决策树(decision tree)属于分类/回归方法.其具有可读性.可解释性.分类速度快等优 ...

  6. smarty使用小技巧——截取小技巧

    smarty截取字符串(末尾没有...)今天发现有个网页出现乱码,检查发现是用truncate()函数截取的字符串,truncate()函数对中文支持不好,随用mb_substr()函数替换trunc ...

  7. 一个漂亮的输出MySql数据库表结构的PHP页面

    经常为了方便和直观,我们会首先直接在数据库中设计出表,但是接下来又要将表的结构和设计编写在设计文档中,以便编码的时候可以直观的查询,一旦数据库表非常多,字段非常多的时候,这无疑是件非常郁闷的工作. 这 ...

  8. Greenwich.SR2版本的Spring Cloud Hystrix实例

    之前我们在eureka(参见Greenwich.SR2版本的Spring Cloud Eureka实例)中,服务消费方a-beautiful-client调用服务提供方a-bootiful-clien ...

  9. Qt编写自定义控件39-导航标签

    一.前言 在很多菜单导航界面中,当单击了二级菜单或者三级菜单以后,顶部会显示带箭头或者其他标识的导航标签,可以单击该标签快速切换到对应的界面,也作为指示当前处于哪一级菜单下的界面,主要在WEB中大肆流 ...

  10. LeetCode_107. Binary Tree Level Order Traversal II

    107. Binary Tree Level Order Traversal II Easy Given a binary tree, return the bottom-up level order ...