赞
踩
设计模式(Design Pattern)是前辈们对代码开发经验的总结,是解决特定问题的一系列套路。它不是语法规定,而是一套用来提高代码可复用性、可维护性、可读性、稳健性以及安全性的解决方案。
1995 年,GoF(Gang of Four,四人组/四人帮)合作出版了《设计模式:可复用面向对象软件的基础》一书,共收录了 23 种设计模式,从此树立了软件设计模式领域的里程碑,人称「GoF设计模式」。
这 23 种设计模式的本质是面向对象设计原则的实际运用,是对类的封装性、继承性和多态性,以及类的关联关系和组合关系的充分理解。
当然,软件设计模式只是一个引导,在实际的软件开发中,必须根据具体的需求来选择:
设计模式有两种分类方法,即根据模式的目的来分和根据模式的作用的范围来分。
1. 根据目的来分
根据模式是用来完成什么工作来划分,这种方式可分为创建型模式、结构型模式和行为型模式 3 种。
2. 根据作用范围来分
根据模式是主要用于类上还是主要用于对象上来分,这种方式可分为类模式和对象模式两种。
3. GoF的23种设计模式的功能
前面说明了 GoF 的 23 种设计模式的分类,现在对各个模式的功能进行介绍。
必须指出,这 23 种设计模式不是孤立存在的,很多模式之间存在一定的关联关系,在大的系统开发中常常同时使用多种设计模式,希望读者认真学好它们。
在有些系统中,为了节省内存资源、保证数据内容的一致性,对某些类要求只能创建一个实例,这就是所谓的单例模式。
创建型模式的主要关注点是“怎样创建对象?”,它的主要特点是“将对象的创建与使用分离”。这样可以降低系统的耦合度,使用者不需要关注对象的创建细节,对象的创建由相关的工厂来完成。就像我们去商场购买商品时,不需要知道商品是怎么生产出来一样,因为它们由专门的厂商生产。
创建型模式分为以下几种。
以上 5 种创建型模式,除了工厂方法模式属于类创建型模式,其他的全部属于对象创建型模式,我们将在之后的教程中详细地介绍它们的特点、结构与应用。
单例(Singleton)模式的定义:指一个类只有一个实例,且该类能自行创建这个实例的一种模式。例如,Windows 中只能打开一个任务管理器,这样可以避免因打开多个任务管理器窗口而造成内存资源的浪费,或出现各个窗口显示内容的不一致等错误。
在计算机系统中,还有 Windows 的回收站、操作系统中的文件系统、多线程中的线程池、显卡的驱动程序对象、打印机的后台处理服务、应用程序的日志对象、数据库的连接池、网站的计数器、Web 应用的配置对象、应用程序中的对话框、系统中的缓存等常常被设计成单例。
单例模式在现实生活中的应用也非常广泛,例如公司 CEO、部门经理等都属于单例模型。J2EE 标准中的 ServletContext 和 ServletContextConfig、Spring 框架应用中的 ApplicationContext、数据库中的连接池等也都是单例模式。
单例模式有 3 个特点:
单例模式的优点:
单例模式的缺点:
单例模式看起来非常简单,实现起来也非常简单。单例模式在面试中是一个高频面试题。希望大家能够认真学习,掌握单例模式,提升核心竞争力,给面试加分,顺利拿到 Offer。
对于Java来说,单例模式可以保证在一个 JVM 中只存在单一实例。单例模式的应用场景主要有以下几个方面。
单例模式是设计模式中最简单的模式之一。通常,普通类的构造函数是公有的,外部类可以通过“new 构造函数()”来生成多个实例。但是,如果将类的构造函数设为私有的,外部类就无法调用该构造函数,也就无法生成多个实例。这时该类自身必须定义一个静态私有实例,并向外提供一个静态的公有函数用于创建或获取该静态私有实例。
下面来分析其基本结构和实现方法。
单例模式的主要角色如下。
其结构如图所示。
Singleton 模式通常有两种实现形式。
第 1 种:懒汉式单例
该模式的特点是类加载时没有生成单例,只有当第一次调用 getlnstance 方法时才去创建这个单例。代码如下:
- public class LazySingleton {
- private static volatile LazySingleton instance = null; //保证 instance 在所有线程中同步
- private LazySingleton() {
- } //private 避免类在外部被实例化
- public static synchronized LazySingleton getInstance() {
- //getInstance 方法前加同步
- if (instance == null) {
- instance = new LazySingleton();
- }
- return instance;
- }
- }
注意:如果编写的是多线程程序,则不要删除上例代码中的关键字 volatile 和 synchronized,否则将存在线程非安全的问题。如果不删除这两个关键字就能保证线程安全,但是每次访问时都要同步,会影响性能,且消耗更多的资源,这是懒汉式单例的缺点。
第 2 种:饿汉式单例
该模式的特点是类一旦加载就创建一个单例,保证在调用 getInstance 方法之前单例已经存在了。
- public class HungrySingleton {
- private static final HungrySingleton instance = new HungrySingleton();
- private HungrySingleton() {
- }
- public static HungrySingleton getInstance() {
- return instance;
- }
- }
饿汉式单例在类创建的同时就已经创建好一个静态的对象供系统使用,以后不再改变,所以是线程安全的,可以直接用于多线程而不会出现问题。
【例1】用懒汉式单例模式模拟产生美国当今总统对象。
分析:在每一届任期内,美国的总统只有一人,所以本实例适合用单例模式实现,图所示是用懒汉式单例实现的结构图。
程序代码如下:
- public class SingletonLazy {
- public static void main(String[] args) {
- President zt1 = President.getInstance();
- zt1.getName(); //输出经理的名字
- President zt2 = President.getInstance();
- zt2.getName(); //输出经理的名字
- if (zt1 == zt2) {
- System.out.println("他们是同一人!");
- } else {
- System.out.println("他们不是同一人!");
- }
- }
- }
- class President {
- private static volatile President instance = null; //保证instance在所有线程中同步
- //private避免类在外部被实例化
- private President() {
- System.out.println("产生一个经理!");
- }
- public static synchronized President getInstance() {
- //在getInstance方法上加同步
- if (instance == null) {
- instance = new President();
- } else {
- System.out.println("已经有一个经理,不能产生新经理!");
- }
- return instance;
- }
- public void getName() {
- System.out.println("我是酒店经理:王小二。");
- }
- }
程序运行结果如下:
产生一个经理! 我是酒店经理:王小二。 已经有一个经理,不能产生新经理! 我是酒店经理:王小二。 他们是同一人!
【例2】用饿汉式单例模式模拟产生猪八戒对象。
分析:同上例类似,猪八戒也只有一个,所以本实例同样适合用单例模式实现。本实例由于要显示猪八戒的图像(点此下载该程序所要显示的猪八戒图片),所以用到了框架窗体 JFrame 组件,这里的猪八戒类是单例类,可以将其定义成面板 JPanel 的子类,里面包含了标签,用于保存猪八戒的图像,客户窗体可以获得猪八戒对象,并显示它。图所示是用饿汉式单例实现的结构图。
程序代码如下:
- import java.awt.*;
- import javax.swing.*;
- public class SingletonEager {
- public static void main(String[] args) {
- JFrame jf = new JFrame("饿汉单例模式测试");
- jf.setLayout(new GridLayout(1, 2));
- Container contentPane = jf.getContentPane();
- Bajie obj1 = Bajie.getInstance();
- contentPane.add(obj1);
- Bajie obj2 = Bajie.getInstance();
- contentPane.add(obj2);
- if (obj1 == obj2) {
- System.out.println("他们是同一人!");
- } else {
- System.out.println("他们不是同一人!");
- }
- jf.pack();
- jf.setVisible(true);
- jf.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
- }
- }
- class Bajie extends JPanel {
- private static Bajie instance = new Bajie();
- private Bajie() {
- JLabel l1 = new JLabel(new ImageIcon("src/Bajie.jpg"));
- this.add(l1);
- }
- public static Bajie getInstance() {
- return instance;
- }
- }
程序运行结果如图所示。
单例模式可扩展为有限的多例(Multitcm)模式,这种模式可生成有限个实例并保存在 ArrayList 中,客户需要时可随机获取,其结构图如图所示。
在有些系统中,存在大量相同或相似对象的创建问题,如果用传统的构造函数来创建对象,会比较复杂且耗时耗资源,用原型模式生成对象就很高效,就像孙悟空拔下猴毛轻轻一吹就变出很多孙悟空一样简单。
原型(Prototype)模式的定义如下:用一个已经创建的实例作为原型,通过复制该原型对象来创建一个和原型相同或相似的新对象。在这里,原型实例指定了要创建的对象的种类。用这种方式创建对象非常高效,根本无须知道对象创建的细节。例如,Windows 操作系统的安装通常较耗时,如果复制就快了很多。在生活中复制的例子非常多,这里不一一列举了。
原型模式的优点:
原型模式的缺点:
由于 Java 提供了对象的 clone() 方法,所以用 Java 实现原型模式很简单。
1. 模式的结构
原型模式包含以下主要角色。
其结构图如图所示。
2. 模式的实现
原型模式的克隆分为浅克隆和深克隆。
Java 中的 Object 类提供了浅克隆的 clone() 方法,具体原型类只要实现 Cloneable 接口就可实现对象的浅克隆,这里的 Cloneable 接口就是抽象原型类。其代码如下:
- //具体原型类
- class Realizetype implements Cloneable {
- Realizetype() {
- System.out.println("具体原型创建成功!");
- }
- public Object clone() throws CloneNotSupportedException {
- System.out.println("具体原型复制成功!");
- return (Realizetype) super.clone();
- }
- }
- //原型模式的测试类
- public class PrototypeTest {
- public static void main(String[] args) throws CloneNotSupportedException {
- Realizetype obj1 = new Realizetype();
- Realizetype obj2 = (Realizetype) obj1.clone();
- System.out.println("obj1==obj2?" + (obj1 == obj2));
- }
- }
程序的运行结果如下:
具体原型创建成功! 具体原型复制成功! obj1==obj2?false
【例1】用原型模式模拟“孙悟空”复制自己。
分析:孙悟空拔下猴毛轻轻一吹就变出很多孙悟空,这实际上是用到了原型模式。这里的孙悟空类 SunWukong 是具体原型类,而 Java 中的 Cloneable 接口是抽象原型类。
同前面介绍的猪八戒实例一样,由于要显示孙悟空的图像(点击此处下载该程序所要显示的孙悟空的图片),所以将孙悟空类定义成面板 JPanel 的子类,里面包含了标签,用于保存孙悟空的图像。
另外,重写了 Cloneable 接口的 clone() 方法,用于复制新的孙悟空。访问类可以通过调用孙悟空的 clone() 方法复制多个孙悟空,并在框架窗体 JFrame 中显示。图所示是其结构图。
程序代码如下:
- import java.awt.*;
- import javax.swing.*;
- class SunWukong extends JPanel implements Cloneable {
- private static final long serialVersionUID = 5543049531872119328L;
- public SunWukong() {
- JLabel l1 = new JLabel(new ImageIcon("src/Wukong.jpg"));
- this.add(l1);
- }
- public Object clone() {
- SunWukong w = null;
- try {
- w = (SunWukong) super.clone();
- } catch (CloneNotSupportedException e) {
- System.out.println("拷贝悟空失败!");
- }
- return w;
- }
- }
- public class ProtoTypeWukong {
- public static void main(String[] args) {
- JFrame jf = new JFrame("原型模式测试");
- jf.setLayout(new GridLayout(1, 2));
- Container contentPane = jf.getContentPane();
- SunWukong obj1 = new SunWukong();
- contentPane.add(obj1);
- SunWukong obj2 = (SunWukong) obj1.clone();
- contentPane.add(obj2);
- jf.pack();
- jf.setVisible(true);
- jf.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
- }
- }
程序的运行结果如图所示。
用原型模式除了可以生成相同的对象,还可以生成相似的对象,请看以下实例。
【例2】用原型模式生成“三好学生”奖状。
分析:同一学校的“三好学生”奖状除了获奖人姓名不同,其他都相同,属于相似对象的复制,同样可以用原型模式创建,然后再做简单修改就可以了。图 4 所示是三好学生奖状生成器的结构图。
程序代码如下:
- public class ProtoTypeCitation {
- public static void main(String[] args) throws CloneNotSupportedException {
- citation obj1 = new citation("张三", "同学:在2016学年第一学期中表现优秀,被评为三好学生。", "韶关学院");
- obj1.display();
- citation obj2 = (citation) obj1.clone();
- obj2.setName("李四");
- obj2.display();
- }
- }
- //奖状类
- class citation implements Cloneable {
- String name;
- String info;
- String college;
- citation(String name, String info, String college) {
- this.name = name;
- this.info = info;
- this.college = college;
- System.out.println("奖状创建成功!");
- }
- void setName(String name) {
- this.name = name;
- }
- String getName() {
- return (this.name);
- }
- void display() {
- System.out.println(name + info + college);
- }
- public Object clone() throws CloneNotSupportedException {
- System.out.println("奖状拷贝成功!");
- return (citation) super.clone();
- }
- }
程序运行结果如下:
奖状创建成功! 张三同学:在2016学年第一学期中表现优秀,被评为三好学生。韶关学院 奖状拷贝成功! 李四同学:在2016学年第一学期中表现优秀,被评为三好学生。韶关学院
原型模式通常适用于以下场景。
在 Spring 中,原型模式应用的非常广泛,例如 scope='prototype'、JSON.parseObject() 等都是原型模式的具体应用。
原型模式可扩展为带原型管理器的原型模式,它在原型模式的基础上增加了一个原型管理器 PrototypeManager 类。该类用 HashMap 保存多个复制的原型,Client 类可以通过管理器的 get(String id) 方法从中获取复制的原型。其结构图如图所示。
【例3】用带原型管理器的原型模式来生成包含“圆”和“正方形”等图形的原型,并计算其面积。分析:本实例中由于存在不同的图形类,例如,“圆”和“正方形”,它们计算面积的方法不一样,所以需要用一个原型管理器来管理它们,图所示是其结构图。
程序代码如下:
- import java.util.*;
- interface Shape extends Cloneable {
- public Object clone(); //拷贝
- public void countArea(); //计算面积
- }
- class Circle implements Shape {
- public Object clone() {
- Circle w = null;
- try {
- w = (Circle) super.clone();
- } catch (CloneNotSupportedException e) {
- System.out.println("拷贝圆失败!");
- }
- return w;
- }
- public void countArea() {
- int r = 0;
- System.out.print("这是一个圆,请输入圆的半径:");
- Scanner input = new Scanner(System.in);
- r = input.nextInt();
- System.out.println("该圆的面积=" + 3.1415 * r * r + "\n");
- }
- }
- class Square implements Shape {
- public Object clone() {
- Square b = null;
- try {
- b = (Square) super.clone();
- } catch (CloneNotSupportedException e) {
- System.out.println("拷贝正方形失败!");
- }
- return b;
- }
- public void countArea() {
- int a = 0;
- System.out.print("这是一个正方形,请输入它的边长:");
- Scanner input = new Scanner(System.in);
- a = input.nextInt();
- System.out.println("该正方形的面积=" + a * a + "\n");
- }
- }
- class ProtoTypeManager {
- private HashMap<String, Shape> ht = new HashMap<String, Shape>();
- public ProtoTypeManager() {
- ht.put("Circle", new Circle());
- ht.put("Square", new Square());
- }
- public void addshape(String key, Shape obj) {
- ht.put(key, obj);
- }
- public Shape getShape(String key) {
- Shape temp = ht.get(key);
- return (Shape) temp.clone();
- }
- }
- public class ProtoTypeShape {
- public static void main(String[] args) {
- ProtoTypeManager pm = new ProtoTypeManager();
- Shape obj1 = (Circle) pm.getShape("Circle");
- obj1.countArea();
- Shape obj2 = (Shape) pm.getShape("Square");
- obj2.countArea();
- }
- }
运行结果如下所示:
这是一个圆,请输入圆的半径:3 该圆的面积=28.2735 这是一个正方形,请输入它的边长:3 该正方形的面积=9
现实生活中,原始社会自给自足(没有工厂),农耕社会小作坊(简单工厂,民间酒坊),工业革命流水线(工厂方法,自产自销),现代产业链代工厂(抽象工厂,富士康)。我们的项目代码同样是由简到繁一步一步迭代而来的,但对于调用者来说,却越来越简单。
在日常开发中,凡是需要生成复杂对象的地方,都可以尝试考虑使用工厂模式来代替。
注意:上述复杂对象指的是类的构造函数参数过多等对类的构造有影响的情况,因为类的构造过于复杂,如果直接在其他业务类内使用,则两者的耦合过重,后续业务更改,就需要在任何引用该类的源代码内进行更改,光是查找所有依赖就很消耗时间了,更别说要一个一个修改了。
工厂模式的定义:定义一个创建产品对象的工厂接口,将产品对象的实际创建工作推迟到具体子工厂类当中。这满足创建型模式中所要求的“创建与使用相分离”的特点。
按实际业务场景划分,工厂模式有 3 种不同的实现方式,分别是简单工厂模式、工厂方法模式和抽象工厂模式。
我们把被创建的对象称为“产品”,把创建产品的对象称为“工厂”。如果要创建的产品不多,只要一个工厂类就可以完成,这种模式叫“简单工厂模式”。
在简单工厂模式中创建实例的方法通常为静态(static)方法,因此简单工厂模式(Simple Factory Pattern)又叫作静态工厂方法模式(Static Factory Method Pattern)。
简单来说,简单工厂模式有一个具体的工厂类,可以生成多个不同的产品,属于创建型设计模式。简单工厂模式不在 GoF 23 种设计模式之列。
简单工厂模式每增加一个产品就要增加一个具体产品类和一个对应的具体工厂类,这增加了系统的复杂度,违背了“开闭原则”。
“工厂方法模式”是对简单工厂模式的进一步抽象化,其好处是可以使系统在不修改原来代码的情况下引进新的产品,即满足开闭原则。
优点:
缺点:
应用场景
对于产品种类相对较少的情况,考虑使用简单工厂模式。使用简单工厂模式的客户端只需要传入工厂类的参数,不需要关心如何创建对象的逻辑,可以很方便地创建所需产品。
简单工厂模式的主要角色如下:
其结构图如下图所示。
图 1 简单工厂模式的结构图
根据上图写出该模式的代码如下:
- public class Client {
- public static void main(String[] args) {
- }
- //抽象产品
- public interface Product {
- void show();
- }
- //具体产品:ProductA
- static class ConcreteProduct1 implements Product {
- public void show() {
- System.out.println("具体产品1显示...");
- }
- }
- //具体产品:ProductB
- static class ConcreteProduct2 implements Product {
- public void show() {
- System.out.println("具体产品2显示...");
- }
- }
- final class Const {
- static final int PRODUCT_A = 0;
- static final int PRODUCT_B = 1;
- static final int PRODUCT_C = 2;
- }
- static class SimpleFactory {
- public static Product makeProduct(int kind) {
- switch (kind) {
- case Const.PRODUCT_A:
- return new ConcreteProduct1();
- case Const.PRODUCT_B:
- return new ConcreteProduct2();
- }
- return null;
- }
- }
- }
在现实生活中社会分工越来越细,越来越专业化。各种产品有专门的工厂生产,彻底告别了自给自足的小农经济时代,这大大缩短了产品的生产周期,提高了生产效率。同样,在软件开发中能否做到软件对象的生产和使用相分离呢?能否在满足“开闭原则”的前提下,客户随意增删或改变对软件相关对象的使用呢?这就是本节要讨论的问题。
在《简单工厂模式》一节我们介绍了简单工厂模式,提到了简单工厂模式违背了开闭原则,而“工厂方法模式”是对简单工厂模式的进一步抽象化,其好处是可以使系统在不修改原来代码的情况下引进新的产品,即满足开闭原则。
优点:
缺点:
应用场景:
工厂方法模式由抽象工厂、具体工厂、抽象产品和具体产品等4个要素构成。本节来分析其基本结构和实现方法。
1. 模式的结构
工厂方法模式的主要角色如下。
其结构图如图 1 所示。
图1 工厂方法模式的结构图
2. 模式的实现
根据图 1 写出该模式的代码如下:
- package FactoryMethod;
- public class AbstractFactoryTest {
- public static void main(String[] args) {
- try {
- Product a;
- AbstractFactory af;
- af = (AbstractFactory) ReadXML1.getObject();
- a = af.newProduct();
- a.show();
- } catch (Exception e) {
- System.out.println(e.getMessage());
- }
- }
- }
- //抽象产品:提供了产品的接口
- interface Product {
- public void show();
- }
- //具体产品1:实现抽象产品中的抽象方法
- class ConcreteProduct1 implements Product {
- public void show() {
- System.out.println("具体产品1显示...");
- }
- }
- //具体产品2:实现抽象产品中的抽象方法
- class ConcreteProduct2 implements Product {
- public void show() {
- System.out.println("具体产品2显示...");
- }
- }
- //抽象工厂:提供了厂品的生成方法
- interface AbstractFactory {
- public Product newProduct();
- }
- //具体工厂1:实现了厂品的生成方法
- class ConcreteFactory1 implements AbstractFactory {
- public Product newProduct() {
- System.out.println("具体工厂1生成-->具体产品1...");
- return new ConcreteProduct1();
- }
- }
- //具体工厂2:实现了厂品的生成方法
- class ConcreteFactory2 implements AbstractFactory {
- public Product newProduct() {
- System.out.println("具体工厂2生成-->具体产品2...");
- return new ConcreteProduct2();
- }
- }
- package FactoryMethod;
- import javax.xml.parsers.*;
- import org.w3c.dom.*;
- import java.io.*;
- class ReadXML1 {
- //该方法用于从XML配置文件中提取具体类类名,并返回一个实例对象
- public static Object getObject() {
- try {
- //创建文档对象
- DocumentBuilderFactory dFactory = DocumentBuilderFactory.newInstance();
- DocumentBuilder builder = dFactory.newDocumentBuilder();
- Document doc;
- doc = builder.parse(new File("src/FactoryMethod/config1.xml"));
- //获取包含类名的文本节点
- NodeList nl = doc.getElementsByTagName("className");
- Node classNode = nl.item(0).getFirstChild();
- String cName = "FactoryMethod." + classNode.getNodeValue();
- //System.out.println("新类名:"+cName);
- //通过类名生成实例对象并将其返回
- Class<?> c = Class.forName(cName);
- Object obj = c.newInstance();
- return obj;
- } catch (Exception e) {
- e.printStackTrace();
- return null;
- }
- }
- }
注意:该程序中用到了 XML 文件,如果想要获取该文件,请点击“下载”,就可以对其进行下载。
程序运行结果如下:
具体工厂1生成-->具体产品1... 具体产品1显示...
如果将 XML 配置文件中的 ConcreteFactory1 改为 ConcreteFactory2,则程序运行结果如下:
具体工厂2生成-->具体产品2... 具体产品2显示...
【例1】用工厂方法模式设计畜牧场。
分析:有很多种类的畜牧场,如养马场用于养马,养牛场用于养牛,所以该实例用工厂方法模式比较适合。
对养马场和养牛场等具体工厂类,只要定义一个生成动物的方法 newAnimal() 即可。由于要显示马类和牛类等具体产品类的图像,所以它们的构造函数中用到了 JPanel、JLabd 和 ImageIcon 等组件,并定义一个 show() 方法来显示它们。
客户端程序通过对象生成器类 ReadXML2 读取 XML 配置文件中的数据来决定养马还是养牛。其结构图如图 2 所示。
图2 畜牧场结构图
注意:该程序中用到了 XML 文件,并且要显示马类和牛类等具体产品类的图像,如果想要获取 HTML 文件和图片,请点击“下载”,就可以对其进行下载。
程序代码如下:
- package FactoryMethod;
- import java.awt.*;
- import javax.swing.*;
- public class AnimalFarmTest {
- public static void main(String[] args) {
- try {
- Animal a;
- AnimalFarm af;
- af = (AnimalFarm) ReadXML2.getObject();
- a = af.newAnimal();
- a.show();
- } catch (Exception e) {
- System.out.println(e.getMessage());
- }
- }
- }
- //抽象产品:动物类
- interface Animal {
- public void show();
- }
- //具体产品:马类
- class Horse implements Animal {
- JScrollPane sp;
- JFrame jf = new JFrame("工厂方法模式测试");
- public Horse() {
- Container contentPane = jf.getContentPane();
- JPanel p1 = new JPanel();
- p1.setLayout(new GridLayout(1, 1));
- p1.setBorder(BorderFactory.createTitledBorder("动物:马"));
- sp = new JScrollPane(p1);
- contentPane.add(sp, BorderLayout.CENTER);
- JLabel l1 = new JLabel(new ImageIcon("src/A_Horse.jpg"));
- p1.add(l1);
- jf.pack();
- jf.setVisible(false);
- jf.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); //用户点击窗口关闭
- }
- public void show() {
- jf.setVisible(true);
- }
- }
- //具体产品:牛类
- class Cattle implements Animal {
- JScrollPane sp;
- JFrame jf = new JFrame("工厂方法模式测试");
- public Cattle() {
- Container contentPane = jf.getContentPane();
- JPanel p1 = new JPanel();
- p1.setLayout(new GridLayout(1, 1));
- p1.setBorder(BorderFactory.createTitledBorder("动物:牛"));
- sp = new JScrollPane(p1);
- contentPane.add(sp, BorderLayout.CENTER);
- JLabel l1 = new JLabel(new ImageIcon("src/A_Cattle.jpg"));
- p1.add(l1);
- jf.pack();
- jf.setVisible(false);
- jf.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); //用户点击窗口关闭
- }
- public void show() {
- jf.setVisible(true);
- }
- }
- //抽象工厂:畜牧场
- interface AnimalFarm {
- public Animal newAnimal();
- }
- //具体工厂:养马场
- class HorseFarm implements AnimalFarm {
- public Animal newAnimal() {
- System.out.println("新马出生!");
- return new Horse();
- }
- }
- //具体工厂:养牛场
- class CattleFarm implements AnimalFarm {
- public Animal newAnimal() {
- System.out.println("新牛出生!");
- return new Cattle();
- }
- }
- package FactoryMethod;
- import javax.xml.parsers.*;
- import org.w3c.dom.*;
- import java.io.*;
- class ReadXML2 {
- public static Object getObject() {
- try {
- DocumentBuilderFactory dFactory = DocumentBuilderFactory.newInstance();
- DocumentBuilder builder = dFactory.newDocumentBuilder();
- Document doc;
- doc = builder.parse(new File("src/FactoryMethod/config2.xml"));
- NodeList nl = doc.getElementsByTagName("className");
- Node classNode = nl.item(0).getFirstChild();
- String cName = "FactoryMethod." + classNode.getNodeValue();
- System.out.println("新类名:" + cName);
- Class<?> c = Class.forName(cName);
- Object obj = c.newInstance();
- return obj;
- } catch (Exception e) {
- e.printStackTrace();
- return null;
- }
- }
- }
程序的运行结果如图 3 所示。
图3 畜牧场养殖的运行结果
注意:当需要生成的产品不多且不会增加,一个具体工厂类就可以完成任务时,可删除抽象工厂类。这时工厂方法模式将退化到简单工厂模式。
前面介绍的工厂方法模式中考虑的是一类产品的生产,如畜牧场只养动物、电视机厂只生产电视机、计算机软件学院只培养计算机软件专业的学生等。
同种类称为同等级,也就是说:工厂方法模式只考虑生产同等级的产品,但是在现实生活中许多工厂是综合型的工厂,能生产多等级(种类) 的产品,如农场里既养动物又种植物,电器厂既生产电视机又生产洗衣机或空调,大学既有软件专业又有生物专业等。
本节要介绍的抽象工厂模式将考虑多等级产品的生产,将同一个具体工厂所生产的位于不同等级的一组产品称为一个产品族,图 1 所示的是海尔工厂和 TCL 工厂所生产的电视机与空调对应的关系图。
图1 电器工厂的产品等级与产品族
抽象工厂(AbstractFactory)模式的定义:是一种为访问类提供一个创建一组相关或相互依赖对象的接口,且访问类无须指定所要产品的具体类就能得到同族的不同等级的产品的模式结构。
抽象工厂模式是工厂方法模式的升级版本,工厂方法模式只生产一个等级的产品,而抽象工厂模式可生产多个等级的产品。
使用抽象工厂模式一般要满足以下条件。
抽象工厂模式除了具有工厂方法模式的优点外,其他主要优点如下。
其缺点是:当产品族中需要增加一个新的产品时,所有的工厂类都需要进行修改。增加了系统的抽象性和理解难度。
抽象工厂模式同工厂方法模式一样,也是由抽象工厂、具体工厂、抽象产品和具体产品等 4 个要素构成,但抽象工厂中方法个数不同,抽象产品的个数也不同。现在我们来分析其基本结构和实现方法。
1. 模式的结构
抽象工厂模式的主要角色如下。
抽象工厂模式的结构图如图 2 所示。
图2 抽象工厂模式的结构图
2. 模式的实现
从图 2 可以看出抽象工厂模式的结构同工厂方法模式的结构相似,不同的是其产品的种类不止一个,所以创建产品的方法也不止一个。下面给出抽象工厂和具体工厂的代码。
(1) 抽象工厂:提供了产品的生成方法。
- interface AbstractFactory {
- public Product1 newProduct1();
- public Product2 newProduct2();
- }
(2) 具体工厂:实现了产品的生成方法。
- class ConcreteFactory1 implements AbstractFactory {
- public Product1 newProduct1() {
- System.out.println("具体工厂 1 生成-->具体产品 11...");
- return new ConcreteProduct11();
- }
- public Product2 newProduct2() {
- System.out.println("具体工厂 1 生成-->具体产品 21...");
- return new ConcreteProduct21();
- }
- }
【例1】用抽象工厂模式设计农场类。
分析:农场中除了像畜牧场一样可以养动物,还可以培养植物,如养马、养牛、种菜、种水果等,所以本实例比前面介绍的畜牧场类复杂,必须用抽象工厂模式来实现。
本例用抽象工厂模式来设计两个农场,一个是韶关农场用于养牛和种菜,一个是上饶农场用于养马和种水果,可以在以上两个农场中定义一个生成动物的方法 newAnimal() 和一个培养植物的方法 newPlant()。
对马类、牛类、蔬菜类和水果类等具体产品类,由于要显示它们的图像(点此下载图片),所以它们的构造函数中用到了 JPanel、JLabel 和 ImageIcon 等组件,并定义一个 show() 方法来显示它们。
客户端程序通过对象生成器类 ReadXML 读取 XML 配置文件中的数据来决定养什么动物和培养什么植物(点此下载 XML 文件)。其结构图如图 3 所示。
图3 农场类的结构图
程序代码如下:
- package AbstractFactory;
- import java.awt.*;
- import javax.swing.*;
- public class FarmTest {
- public static void main(String[] args) {
- try {
- Farm f;
- Animal a;
- Plant p;
- f = (Farm) ReadXML.getObject();
- a = f.newAnimal();
- p = f.newPlant();
- a.show();
- p.show();
- } catch (Exception e) {
- System.out.println(e.getMessage());
- }
- }
- }
- //抽象产品:动物类
- interface Animal {
- public void show();
- }
- //具体产品:马类
- class Horse implements Animal {
- JScrollPane sp;
- JFrame jf = new JFrame("抽象工厂模式测试");
- public Horse() {
- Container contentPane = jf.getContentPane();
- JPanel p1 = new JPanel();
- p1.setLayout(new GridLayout(1, 1));
- p1.setBorder(BorderFactory.createTitledBorder("动物:马"));
- sp = new JScrollPane(p1);
- contentPane.add(sp, BorderLayout.CENTER);
- JLabel l1 = new JLabel(new ImageIcon("src/A_Horse.jpg"));
- p1.add(l1);
- jf.pack();
- jf.setVisible(false);
- jf.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);//用户点击窗口关闭
- }
- public void show() {
- jf.setVisible(true);
- }
- }
- //具体产品:牛类
- class Cattle implements Animal {
- JScrollPane sp;
- JFrame jf = new JFrame("抽象工厂模式测试");
- public Cattle() {
- Container contentPane = jf.getContentPane();
- JPanel p1 = new JPanel();
- p1.setLayout(new GridLayout(1, 1));
- p1.setBorder(BorderFactory.createTitledBorder("动物:牛"));
- sp = new JScrollPane(p1);
- contentPane.add(sp, BorderLayout.CENTER);
- JLabel l1 = new JLabel(new ImageIcon("src/A_Cattle.jpg"));
- p1.add(l1);
- jf.pack();
- jf.setVisible(false);
- jf.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);//用户点击窗口关闭
- }
- public void show() {
- jf.setVisible(true);
- }
- }
- //抽象产品:植物类
- interface Plant {
- public void show();
- }
- //具体产品:水果类
- class Fruitage implements Plant {
- JScrollPane sp;
- JFrame jf = new JFrame("抽象工厂模式测试");
- public Fruitage() {
- Container contentPane = jf.getContentPane();
- JPanel p1 = new JPanel();
- p1.setLayout(new GridLayout(1, 1));
- p1.setBorder(BorderFactory.createTitledBorder("植物:水果"));
- sp = new JScrollPane(p1);
- contentPane.add(sp, BorderLayout.CENTER);
- JLabel l1 = new JLabel(new ImageIcon("src/P_Fruitage.jpg"));
- p1.add(l1);
- jf.pack();
- jf.setVisible(false);
- jf.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);//用户点击窗口关闭
- }
- public void show() {
- jf.setVisible(true);
- }
- }
- //具体产品:蔬菜类
- class Vegetables implements Plant {
- JScrollPane sp;
- JFrame jf = new JFrame("抽象工厂模式测试");
- public Vegetables() {
- Container contentPane = jf.getContentPane();
- JPanel p1 = new JPanel();
- p1.setLayout(new GridLayout(1, 1));
- p1.setBorder(BorderFactory.createTitledBorder("植物:蔬菜"));
- sp = new JScrollPane(p1);
- contentPane.add(sp, BorderLayout.CENTER);
- JLabel l1 = new JLabel(new ImageIcon("src/P_Vegetables.jpg"));
- p1.add(l1);
- jf.pack();
- jf.setVisible(false);
- jf.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);//用户点击窗口关闭
- }
- public void show() {
- jf.setVisible(true);
- }
- }
- //抽象工厂:农场类
- interface Farm {
- public Animal newAnimal();
- public Plant newPlant();
- }
- //具体工厂:韶关农场类
- class SGfarm implements Farm {
- public Animal newAnimal() {
- System.out.println("新牛出生!");
- return new Cattle();
- }
- public Plant newPlant() {
- System.out.println("蔬菜长成!");
- return new Vegetables();
- }
- }
- //具体工厂:上饶农场类
- class SRfarm implements Farm {
- public Animal newAnimal() {
- System.out.println("新马出生!");
- return new Horse();
- }
- public Plant newPlant() {
- System.out.println("水果长成!");
- return new Fruitage();
- }
- }
- package AbstractFactory;
- import javax.xml.parsers.*;
- import org.w3c.dom.*;
- import java.io.*;
- class ReadXML {
- public static Object getObject() {
- try {
- DocumentBuilderFactory dFactory = DocumentBuilderFactory.newInstance();
- DocumentBuilder builder = dFactory.newDocumentBuilder();
- Document doc;
- doc = builder.parse(new File("src/AbstractFactory/config.xml"));
- NodeList nl = doc.getElementsByTagName("className");
- Node classNode = nl.item(0).getFirstChild();
- String cName = "AbstractFactory." + classNode.getNodeValue();
- System.out.println("新类名:" + cName);
- Class<?> c = Class.forName(cName);
- Object obj = c.newInstance();
- return obj;
- } catch (Exception e) {
- e.printStackTrace();
- return null;
- }
- }
- }
程序运行结果如图 4 所示。
图4 农场养殖的运行结果
抽象工厂模式最早的应用是用于创建属于不同操作系统的视窗构件。如 Java 的 AWT 中的 Button 和 Text 等构件在 Windows 和 UNIX 中的本地实现是不同的。
抽象工厂模式通常适用于以下场景:
抽象工厂模式的扩展有一定的“开闭原则”倾斜性:
另一方面,当系统中只存在一个等级结构的产品时,抽象工厂模式将退化到工厂方法模式。
在软件开发过程中有时需要创建一个复杂的对象,这个复杂对象通常由多个子部件按一定的步骤组合而成。例如,计算机是由 CPU、主板、内存、硬盘、显卡、机箱、显示器、键盘、鼠标等部件组装而成的,采购员不可能自己去组装计算机,而是将计算机的配置要求告诉计算机销售公司,计算机销售公司安排技术人员去组装计算机,然后再交给要买计算机的采购员。
生活中这样的例子很多,如游戏中的不同角色,其性别、个性、能力、脸型、体型、服装、发型等特性都有所差异;还有汽车中的方向盘、发动机、车架、轮胎等部件也多种多样;每封电子邮件的发件人、收件人、主题、内容、附件等内容也各不相同。
以上所有这些产品都是由多个部件构成的,各个部件可以灵活选择,但其创建步骤都大同小异。这类产品的创建无法用前面介绍的工厂模式描述,只有建造者模式可以很好地描述该类产品的创建。
建造者(Builder)模式的定义:指将一个复杂对象的构造与它的表示分离,使同样的构建过程可以创建不同的表示,这样的设计模式被称为建造者模式。它是将一个复杂的对象分解为多个简单的对象,然后一步一步构建而成。它将变与不变相分离,即产品的组成部分是不变的,但每一部分是可以灵活选择的。
该模式的主要优点如下:
其缺点如下:
建造者(Builder)模式和工厂模式的关注点不同:建造者模式注重零部件的组装过程,而工厂方法模式更注重零部件的创建过程,但两者可以结合使用。
建造者(Builder)模式由产品、抽象建造者、具体建造者、指挥者等 4 个要素构成,现在我们来分析其基本结构和实现方法。
1. 模式的结构
建造者(Builder)模式的主要角色如下。
其结构图如图 1 所示。
图1 建造者模式的结构图
2. 模式的实现
图 1 给出了建造者(Builder)模式的主要结构,其相关类的代码如下。
(1) 产品角色:包含多个组成部件的复杂对象。
- class Product {
- private String partA;
- private String partB;
- private String partC;
- public void setPartA(String partA) {
- this.partA = partA;
- }
- public void setPartB(String partB) {
- this.partB = partB;
- }
- public void setPartC(String partC) {
- this.partC = partC;
- }
- public void show() {
- //显示产品的特性
- }
- }
(2) 抽象建造者:包含创建产品各个子部件的抽象方法。
- abstract class Builder {
- //创建产品对象
- protected Product product = new Product();
- public abstract void buildPartA();
- public abstract void buildPartB();
- public abstract void buildPartC();
- //返回产品对象
- public Product getResult() {
- return product;
- }
- }
(3) 具体建造者:实现了抽象建造者接口。
- public class ConcreteBuilder extends Builder {
- public void buildPartA() {
- product.setPartA("建造 PartA");
- }
- public void buildPartB() {
- product.setPartB("建造 PartB");
- }
- public void buildPartC() {
- product.setPartC("建造 PartC");
- }
- }
(4) 指挥者:调用建造者中的方法完成复杂对象的创建。
- class Director {
- private Builder builder;
- public Director(Builder builder) {
- this.builder = builder;
- }
- //产品构建与组装方法
- public Product construct() {
- builder.buildPartA();
- builder.buildPartB();
- builder.buildPartC();
- return builder.getResult();
- }
- }
(5) 客户类。
- public class Client {
- public static void main(String[] args) {
- Builder builder = new ConcreteBuilder();
- Director director = new Director(builder);
- Product product = director.construct();
- product.show();
- }
- }
【例1】用建造者(Builder)模式描述客厅装修。
分析:客厅装修是一个复杂的过程,它包含墙体的装修、电视机的选择、沙发的购买与布局等。客户把装修要求告诉项目经理,项目经理指挥装修工人一步步装修,最后完成整个客厅的装修与布局,所以本实例用建造者模式实现比较适合。
这里客厅是产品,包括墙、电视和沙发等组成部分。具体装修工人是具体建造者,他们负责装修与墙、电视和沙发的布局。项目经理是指挥者,他负责指挥装修工人进行装修。
另外,客厅类中提供了 show() 方法,可以将装修效果图显示出来(点此下载装修效果图的图片)。客户端程序通过对象生成器类 ReadXML 读取 XML 配置文件中的装修方案数据(点此下载 XML 文件),调用项目经理进行装修。其类图如图 2 所示。
图2 客厅装修的结构图
程序代码如下:
- package Builder;
- import java.awt.*;
- import javax.swing.*;
- public class ParlourDecorator {
- public static void main(String[] args) {
- try {
- Decorator d;
- d = (Decorator) ReadXML.getObject();
- ProjectManager m = new ProjectManager(d);
- Parlour p = m.decorate();
- p.show();
- } catch (Exception e) {
- System.out.println(e.getMessage());
- }
- }
- }
- //产品:客厅
- class Parlour {
- private String wall; //墙
- private String TV; //电视
- private String sofa; //沙发
- public void setWall(String wall) {
- this.wall = wall;
- }
- public void setTV(String TV) {
- this.TV = TV;
- }
- public void setSofa(String sofa) {
- this.sofa = sofa;
- }
- public void show() {
- JFrame jf = new JFrame("建造者模式测试");
- Container contentPane = jf.getContentPane();
- JPanel p = new JPanel();
- JScrollPane sp = new JScrollPane(p);
- String parlour = wall + TV + sofa;
- JLabel l = new JLabel(new ImageIcon("src/" + parlour + ".jpg"));
- p.setLayout(new GridLayout(1, 1));
- p.setBorder(BorderFactory.createTitledBorder("客厅"));
- p.add(l);
- contentPane.add(sp, BorderLayout.CENTER);
- jf.pack();
- jf.setVisible(true);
- jf.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
- }
- }
- //抽象建造者:装修工人
- abstract class Decorator {
- //创建产品对象
- protected Parlour product = new Parlour();
- public abstract void buildWall();
- public abstract void buildTV();
- public abstract void buildSofa();
- //返回产品对象
- public Parlour getResult() {
- return product;
- }
- }
- //具体建造者:具体装修工人1
- class ConcreteDecorator1 extends Decorator {
- public void buildWall() {
- product.setWall("w1");
- }
- public void buildTV() {
- product.setTV("TV1");
- }
- public void buildSofa() {
- product.setSofa("sf1");
- }
- }
- //具体建造者:具体装修工人2
- class ConcreteDecorator2 extends Decorator {
- public void buildWall() {
- product.setWall("w2");
- }
- public void buildTV() {
- product.setTV("TV2");
- }
- public void buildSofa() {
- product.setSofa("sf2");
- }
- }
- //指挥者:项目经理
- class ProjectManager {
- private Decorator builder;
- public ProjectManager(Decorator builder) {
- this.builder = builder;
- }
- //产品构建与组装方法
- public Parlour decorate() {
- builder.buildWall();
- builder.buildTV();
- builder.buildSofa();
- return builder.getResult();
- }
- }
- package Builder;
- import org.w3c.dom.Document;
- import org.w3c.dom.Node;
- import org.w3c.dom.NodeList;
- import javax.xml.parsers.DocumentBuilder;
- import javax.xml.parsers.DocumentBuilderFactory;
- import java.io.File;
- class ReadXML {
- public static Object getObject() {
- try {
- DocumentBuilderFactory dFactory = DocumentBuilderFactory.newInstance();
- DocumentBuilder builder = dFactory.newDocumentBuilder();
- Document doc;
- doc = builder.parse(new File("src/Builder/config.xml"));
- NodeList nl = doc.getElementsByTagName("className");
- Node classNode = nl.item(0).getFirstChild();
- String cName = "Builder." + classNode.getNodeValue();
- System.out.println("新类名:" + cName);
- Class<?> c = Class.forName(cName);
- Object obj = c.newInstance();
- return obj;
- } catch (Exception e) {
- e.printStackTrace();
- return null;
- }
- }
- }
程序运行结果如图 3 所示。
图3 客厅装修的运行结果
建造者模式唯一区别于工厂模式的是针对复杂对象的创建。也就是说,如果创建简单对象,通常都是使用工厂模式进行创建,而如果创建复杂对象,就可以考虑使用建造者模式。
当需要创建的产品具备复杂创建过程时,可以抽取出共性创建过程,然后交由具体实现类自定义创建流程,使得同样的创建行为可以生产出不同的产品,分离了创建与表示,使创建产品的灵活性大大增加。
建造者模式主要适用于以下应用场景:
通过前面的学习,我们已经了解了建造者模式,那么它和工厂模式有什么区别呢?
建造者(Builder)模式在应用过程中可以根据需要改变,如果创建的产品种类只有一种,只需要一个具体建造者,这时可以省略掉抽象建造者,甚至可以省略掉指挥者角色。
创建型模式(Creational Pattern)的主要特点是将对象的创建与使用分离,根据对象的创建与组合方式的不同,创建型模式可分为单例(Singleton)模式、原型(Prototype)模式、工厂方法(Factory Method)模式、抽象工厂(Abstract Factory)模式和建造者(Builder)模式 5 种。
本实验的主要目的如下。
1. 创建型模式的工作原理
创建型模式隐藏了对象的创建细节,对象的创建由相关的工厂来完成,使用者不需要关注对象的创建细节,这样可以降低系统的耦合度。
创建型模式共 5 种,它们分别是单例模式、原型模式、工厂方法模式、抽象工厂模式和建造者模式,其工作原理在各自的教程中也都有详细的介绍,每种模式的实验大概要花 2 个学时,大家可以根据实验计划来选做若干个实验,下面以工厂方法模式为例,介绍其实验过程。
2. 工厂方法模式的工作原理
工厂方法模式(Factory Method Pattern),也叫虚拟构造器(Virtual Constructor)模式或者多态工厂(Polymorphic Factory)模式。
在工厂方法模式中,工厂父类负责定义创建产品对象的公共接口,而工厂子类则负责生成具体的产品对象,这样做的目的是将产品类的实例化操作延迟到工厂子类中完成。其结构图如图 1 所示。
图1 工厂方法模式的类图
工厂方法模式包含如下角色。
工厂方法模式的特点是当系统扩展需要添加新的产品对象时,仅仅需要添加一个具体产品对象以及一个具体工厂对象,原有工厂对象不需要进行任何修改,也不需要修改客户端,很好地符合了“开闭原则”。
(1) 用工厂方法模式设计一个电动自行车工厂的模拟程序。
要求:要为每种品牌的电动自行车提供一个子工厂,如爱玛工厂专门负责生产爱玛(Aima)牌电动自行车,雅迪工厂专门负责生产雅迪(Yadea)牌电动自行车。如果今后需要生产台铃(Tailg)牌电动自行车,只需要增加一个新的台铃电动自行车工厂即可,无须修改原有代码,使得整个系统具有更强的灵活性和可扩展性。(点此下载程序运行后所要显示的图片)。
(2) 按照以上要求设计类图和编写 Java 源程序。
所设计的实验程序要满足以下两点。
(1) 用 UML 设计“电动自行车工厂模拟程序”的结构图。“电动自行车工厂模拟程序”的结构图如图 2 所示。
图2 电动自行车工厂模拟程序的结构图
(2) 根据结构图写出“电动自行车工厂模拟程序”的源代码。
① 电动自行车工厂模拟程序的源代码如下。
- package FactoryMethod;
- import java.awt.*;
- import javax.swing.*;
- public class BicycleFactoryTest {
- public static void main(String[] args) {
- try {
- Bicycle a;
- BicycleFactory bf;
- bf = (BicycleFactory) ReadXML.getObject();
- a = bf.produce();
- a.show();
- } catch (Exception e) {
- System.out.println(e.getMessage());
- }
- }
- }
- //抽象产品:自行车
- interface Bicycle {
- public void show();
- }
- //具体产品:爱玛自行车
- class AimaBicycle implements Bicycle {
- JScrollPane sp;
- JFrame jf = new JFrame("工厂方法模式测试");
- public AimaBicycle() {
- JPanel p1 = new JPanel();
- p1.setLayout(new GridLayout(1, 1));
- p1.setBorder(BorderFactory.createTitledBorder("爱玛自行车"));
- JLabel l1 = new JLabel(new ImageIcon("src/FactoryMethod/AIMABicycle.jpg"));
- p1.add(l1);
- sp = new JScrollPane(p1);
- Container contentPane = jf.getContentPane();
- contentPane.add(sp, BorderLayout.CENTER);
- jf.pack();
- jf.setVisible(false);
- jf.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); //用户点击窗口关闭
- }
- public void show() {
- jf.setVisible(true);
- }
- }
- //具体产品:雅迪自行车
- class YadeaBicycle implements Bicycle {
- JScrollPane sp;
- JFrame jf = new JFrame("工厂方法模式测试");
- public YadeaBicycle() {
- JPanel p1 = new JPanel();
- p1.setLayout(new GridLayout(1, 1));
- p1.setBorder(BorderFactory.createTitledBorder("雅迪自行车"));
- JLabel l1 = new JLabel(new ImageIcon("src/FactoryMethod/YadeaBicycle.jpg"));
- p1.add(l1);
- Container contentPane = jf.getContentPane();
- sp = new JScrollPane(p1);
- contentPane.add(sp, BorderLayout.CENTER);
- jf.pack();
- jf.setVisible(false);
- jf.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); //用户点击窗口关闭
- }
- public void show() {
- jf.setVisible(true);
- }
- }
- //抽象工厂:自行车工厂
- interface BicycleFactory {
- public Bicycle produce();
- }
- //具体工厂:爱玛工厂
- class AimaFactory implements BicycleFactory {
- public Bicycle produce() {
- System.out.println("爱玛自行车生产了!");
- return new AimaBicycle();
- }
- }
- //具体工厂:雅迪工厂
- class YadeaFactory implements BicycleFactory {
- public Bicycle produce() {
- System.out.println("雅迪自行车生产了!");
- return new YadeaBicycle();
- }
- }
② 对象生成器的源代码如下。
- package FactoryMethod;
- import org.w3c.dom.Document;
- import org.w3c.dom.Node;
- import org.w3c.dom.NodeList;
- import javax.xml.parsers.DocumentBuilder;
- import javax.xml.parsers.DocumentBuilderFactory;
- import java.io.File;
- class ReadXML {
- public static Object getObject() {
- try {
- DocumentBuilderFactory dFactory = DocumentBuilderFactory.newInstance();
- DocumentBuilder builder = dFactory.newDocumentBuilder();
- Document doc;
- doc = builder.parse(new File("src/FactoryMethod/config.xml"));
- NodeList nl = doc.getElementsByTagName("className");
- Node classNode = nl.item(0).getFirstChild();
- String cName = "FactoryMethod." + classNode.getNodeValue();
- System.out.println("新类名:" + cName);
- Class<?> c = Class.forName(cName);
- Object obj = c.newInstance();
- return obj;
- } catch (Exception e) {
- e.printStackTrace();
- return null;
- }
- }
- }
③XML 配置文件的代码如下。
- <?xml version="1.0" encoding="UTF-8"?>
- <config>
- <className>AimaFactory</className>
- </config>
(3) 上机测试程序,查看运行结果。
“电动自行车工厂模拟程序”的运行结果如图 3 所示。
图3 电动自行车工厂模拟程序的运行结果
(4) 按同样的步骤设计其他“创建型模式”的程序实例。
结构型模式描述如何将类或对象按某种布局组成更大的结构。它分为类结构型模式和对象结构型模式,前者采用继承机制来组织接口和类,后者釆用组合或聚合来组合对象。
由于组合关系或聚合关系比继承关系耦合度低,满足“合成复用原则”,所以对象结构型模式比类结构型模式具有更大的灵活性。
结构型模式分为以下 7 种:
以上 7 种结构型模式,除了适配器模式分为类结构型模式和对象结构型模式两种,其他的全部属于对象结构型模式,下面我们会分别、详细地介绍它们的特点、结构与应用。
在有些情况下,一个客户不能或者不想直接访问另一个对象,这时需要找一个中介帮忙完成某项任务,这个中介就是代理对象。例如,购买火车票不一定要去火车站买,可以通过 12306 网站或者去火车票代售点买。又如找女朋友、找保姆、找工作等都可以通过找中介完成。
在软件设计中,使用代理模式的例子也很多,例如,要访问的远程对象比较大(如视频或大图像等),其下载要花很多时间。还有因为安全原因需要屏蔽客户端直接访问真实对象,如某单位的内部数据库等。
代理模式的定义:由于某些原因需要给某对象提供一个代理以控制对该对象的访问。这时,访问对象不适合或者不能直接引用目标对象,代理对象作为访问对象和目标对象之间的中介。
代理模式的主要优点有:
其主要缺点是:
那么如何解决以上提到的缺点呢?答案是可以使用动态代理方式
代理模式的结构比较简单,主要是通过定义一个继承抽象主题的代理来包含真实主题,从而实现对真实主题的访问,下面来分析其基本结构和实现方法。
1. 模式的结构
代理模式的主要角色如下。
其结构图如图 1 所示。
图1 代理模式的结构图
在代码中,一般代理会被理解为代码增强,实际上就是在原代码逻辑前后增加一些代码逻辑,而使调用者无感知。
根据代理的创建时期,代理模式分为静态代理和动态代理。
2. 模式的实现
代理模式的实现代码如下:
- package proxy;
- public class ProxyTest {
- public static void main(String[] args) {
- Proxy proxy = new Proxy();
- proxy.Request();
- }
- }
- //抽象主题
- interface Subject {
- void Request();
- }
- //真实主题
- class RealSubject implements Subject {
- public void Request() {
- System.out.println("访问真实主题方法...");
- }
- }
- //代理
- class Proxy implements Subject {
- private RealSubject realSubject;
- public void Request() {
- if (realSubject == null) {
- realSubject = new RealSubject();
- }
- preRequest();
- realSubject.Request();
- postRequest();
- }
- public void preRequest() {
- System.out.println("访问真实主题之前的预处理。");
- }
- public void postRequest() {
- System.out.println("访问真实主题之后的后续处理。");
- }
- }
程序运行的结果如下:
访问真实主题之前的预处理。 访问真实主题方法... 访问真实主题之后的后续处理。
【例1】韶关“天街e角”公司是一家婺源特产公司的代理公司,用代理模式实现。
分析:本实例中的“婺源特产公司”经营许多婺源特产,它是真实主题,提供了显示特产的 display() 方法,可以用窗体程序实现(点此下载该实例所要显示的图片)。而韶关“天街e角”公司是婺源特产公司特产的代理,通过调用婺源特产公司的 display() 方法显示代理产品,当然它可以增加一些额外的处理,如包裝或加价等。客户可通过“天街e角”代理公司间接访问“婺源特产公司”的产品,图 2 所示是公司的结构图。
图2 韶关“天街e角”公司的结构图
程序代码如下:
- package proxy;
- import java.awt.*;
- import javax.swing.*;
- public class WySpecialtyProxy {
- public static void main(String[] args) {
- SgProxy proxy = new SgProxy();
- proxy.display();
- }
- }
- //抽象主题:特产
- interface Specialty {
- void display();
- }
- //真实主题:婺源特产
- class WySpecialty extends JFrame implements Specialty {
- private static final long serialVersionUID = 1L;
- public WySpecialty() {
- super("韶关代理婺源特产测试");
- this.setLayout(new GridLayout(1, 1));
- JLabel l1 = new JLabel(new ImageIcon("src/proxy/WuyuanSpecialty.jpg"));
- this.add(l1);
- this.pack();
- this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
- }
- public void display() {
- this.setVisible(true);
- }
- }
- //代理:韶关代理
- class SgProxy implements Specialty {
- private WySpecialty realSubject = new WySpecialty();
- public void display() {
- preRequest();
- realSubject.display();
- postRequest();
- }
- public void preRequest() {
- System.out.println("韶关代理婺源特产开始。");
- }
- public void postRequest() {
- System.out.println("韶关代理婺源特产结束。");
- }
- }
程序运行结果如图 3 所示。
图3 韶关“天街e角”公司的代理产品
当无法或不想直接引用某个对象或访问某个对象存在困难时,可以通过代理对象来间接访问。使用代理模式主要有两个目的:一是保护目标对象,二是增强目标对象。
前面分析了代理模式的结构与特点,现在来分析以下的应用场景。
在前面介绍的代理模式中,代理类中包含了对真实主题的引用,这种方式存在两个缺点。
图4 动态代理模式的结构图
在现实生活中,经常出现两个对象因接口不兼容而不能在一起工作的实例,这时需要第三者进行适配。例如,讲中文的人同讲英文的人对话时需要一个翻译,用直流电的笔记本电脑接交流电源时需要一个电源适配器,用计算机访问照相机的 SD 内存卡时需要一个读卡器等。
在软件设计中也可能出现:需要开发的具有某种业务功能的组件在现有的组件库中已经存在,但它们与当前系统的接口规范不兼容,如果重新开发这些组件成本又很高,这时用适配器模式能很好地解决这些问题。
适配器模式(Adapter)的定义如下:将一个类的接口转换成客户希望的另外一个接口,使得原本由于接口不兼容而不能一起工作的那些类能一起工作。适配器模式分为类结构型模式和对象结构型模式两种,前者类之间的耦合度比后者高,且要求程序员了解现有组件库中的相关组件的内部结构,所以应用相对较少些。
该模式的主要优点如下。
其缺点是:
类适配器模式可采用多重继承方式实现,如 C++ 可定义一个适配器类来同时继承当前系统的业务接口和现有组件库中已经存在的组件接口;Java 不支持多继承,但可以定义一个适配器类来实现当前系统的业务接口,同时又继承现有组件库中已经存在的组件。
对象适配器模式可釆用将现有组件库中已经实现的组件引入适配器类中,该类同时实现当前系统的业务接口。现在来介绍它们的基本结构。
1. 模式的结构
适配器模式(Adapter)包含以下主要角色。
类适配器模式的结构图如图 1 所示。
图1 类适配器模式的结构图
对象适配器模式的结构图如图 2 所示。
图2 对象适配器模式的结构图
2. 模式的实现
(1) 类适配器模式的代码如下。
- package adapter;
- //目标接口
- interface Target
- {
- public void request();
- }
- //适配者接口
- class Adaptee
- {
- public void specificRequest()
- {
- System.out.println("适配者中的业务代码被调用!");
- }
- }
- //类适配器类
- class ClassAdapter extends Adaptee implements Target
- {
- public void request()
- {
- specificRequest();
- }
- }
- //客户端代码
- public class ClassAdapterTest
- {
- public static void main(String[] args)
- {
- System.out.println("类适配器模式测试:");
- Target target = new ClassAdapter();
- target.request();
- }
- }
程序的运行结果如下:
类适配器模式测试: 适配者中的业务代码被调用!
(2)对象适配器模式的代码如下。
- package adapter;
- //对象适配器类
- class ObjectAdapter implements Target
- {
- private Adaptee adaptee;
- public ObjectAdapter(Adaptee adaptee)
- {
- this.adaptee=adaptee;
- }
- public void request()
- {
- adaptee.specificRequest();
- }
- }
- //客户端代码
- public class ObjectAdapterTest
- {
- public static void main(String[] args)
- {
- System.out.println("对象适配器模式测试:");
- Adaptee adaptee = new Adaptee();
- Target target = new ObjectAdapter(adaptee);
- target.request();
- }
- }
说明:对象适配器模式中的“目标接口”和“适配者类”的代码同类适配器模式一样,只要修改适配器类和客户端的代码即可。
程序的运行结果如下:
对象适配器模式测试: 适配者中的业务代码被调用!
【例1】用适配器模式(Adapter)模拟新能源汽车的发动机。
分析:新能源汽车的发动机有电能发动机(Electric Motor)和光能发动机(Optical Motor)等,各种发动机的驱动方法不同,例如,电能发动机的驱动方法 electricDrive() 是用电能驱动,而光能发动机的驱动方法 opticalDrive() 是用光能驱动,它们是适配器模式中被访问的适配者。
客户端希望用统一的发动机驱动方法 drive() 访问这两种发动机,所以必须定义一个统一的目标接口 Motor,然后再定义电能适配器(Electric Adapter)和光能适配器(Optical Adapter)去适配这两种发动机。
我们把客户端想访问的新能源发动机的适配器的名称放在 XML 配置文件中(点此下载 XML 文件),客户端可以通过对象生成器类 ReadXML 去读取。这样,客户端就可以通过 Motor 接口随便使用任意一种新能源发动机去驱动汽车,图 3 所示是其结构图。
图3 发动机适配器的结构图
程序代码如下:
- package adapter;
- //目标:发动机
- interface Motor
- {
- public void drive();
- }
- //适配者1:电能发动机
- class ElectricMotor
- {
- public void electricDrive()
- {
- System.out.println("电能发动机驱动汽车!");
- }
- }
- //适配者2:光能发动机
- class OpticalMotor
- {
- public void opticalDrive()
- {
- System.out.println("光能发动机驱动汽车!");
- }
- }
- //电能适配器
- class ElectricAdapter implements Motor
- {
- private ElectricMotor emotor;
- public ElectricAdapter()
- {
- emotor=new ElectricMotor();
- }
- public void drive()
- {
- emotor.electricDrive();
- }
- }
- //光能适配器
- class OpticalAdapter implements Motor
- {
- private OpticalMotor omotor;
- public OpticalAdapter()
- {
- omotor=new OpticalMotor();
- }
- public void drive()
- {
- omotor.opticalDrive();
- }
- }
- //客户端代码
- public class MotorAdapterTest
- {
- public static void main(String[] args)
- {
- System.out.println("适配器模式测试:");
- Motor motor=(Motor)ReadXML.getObject();
- motor.drive();
- }
- }
- package adapter;
- import javax.xml.parsers.*;
- import org.w3c.dom.*;
- import java.io.*;
- class ReadXML
- {
- public static Object getObject()
- {
- try
- {
- DocumentBuilderFactory dFactory=DocumentBuilderFactory.newInstance();
- DocumentBuilder builder=dFactory.newDocumentBuilder();
- Document doc;
- doc=builder.parse(new File("src/adapter/config.xml"));
- NodeList nl=doc.getElementsByTagName("className");
- Node classNode=nl.item(0).getFirstChild();
- String cName="adapter."+classNode.getNodeValue();
- Class<?> c=Class.forName(cName);
- Object obj=c.newInstance();
- return obj;
- }
- catch(Exception e)
- {
- e.printStackTrace();
- return null;
- }
- }
- }
程序的运行结果如下:
适配器模式测试: 电能发动机驱动汽车!
注意:如果将配置文件中的 ElectricAdapter 改为 OpticalAdapter,则运行结果如下:
适配器模式测试: 光能发动机驱动汽车!
适配器模式(Adapter)通常适用于以下场景。
适配器模式(Adapter)可扩展为双向适配器模式,双向适配器类既可以把适配者接口转换成目标接口,也可以把目标接口转换成适配者接口,其结构图如图 4 所示。
图4 双向适配器模式的结构图
程序代码如下:
- package adapter;
- //目标接口
- interface TwoWayTarget
- {
- public void request();
- }
- //适配者接口
- interface TwoWayAdaptee
- {
- public void specificRequest();
- }
- //目标实现
- class TargetRealize implements TwoWayTarget
- {
- public void request()
- {
- System.out.println("目标代码被调用!");
- }
- }
- //适配者实现
- class AdapteeRealize implements TwoWayAdaptee
- {
- public void specificRequest()
- {
- System.out.println("适配者代码被调用!");
- }
- }
- //双向适配器
- class TwoWayAdapter implements TwoWayTarget,TwoWayAdaptee
- {
- private TwoWayTarget target;
- private TwoWayAdaptee adaptee;
- public TwoWayAdapter(TwoWayTarget target)
- {
- this.target=target;
- }
- public TwoWayAdapter(TwoWayAdaptee adaptee)
- {
- this.adaptee=adaptee;
- }
- public void request()
- {
- adaptee.specificRequest();
- }
- public void specificRequest()
- {
- target.request();
- }
- }
- //客户端代码
- public class TwoWayAdapterTest
- {
- public static void main(String[] args)
- {
- System.out.println("目标通过双向适配器访问适配者:");
- TwoWayAdaptee adaptee=new AdapteeRealize();
- TwoWayTarget target=new TwoWayAdapter(adaptee);
- target.request();
- System.out.println("-------------------");
- System.out.println("适配者通过双向适配器访问目标:");
- target=new TargetRealize();
- adaptee=new TwoWayAdapter(target);
- adaptee.specificRequest();
- }
- }
程序的运行结果如下:
目标通过双向适配器访问适配者: 适配者代码被调用! ------------------- 适配者通过双向适配器访问目标: 目标代码被调用!
在现实生活中,某些类具有两个或多个维度的变化,如图形既可按形状分,又可按颜色分。如何设计类似于 Photoshop 这样的软件,能画不同形状和不同颜色的图形呢?如果用继承方式,m 种形状和 n 种颜色的图形就有 m×n 种,不但对应的子类很多,而且扩展困难。
当然,这样的例子还有很多,如不同颜色和字体的文字、不同品牌和功率的汽车、不同性别和职业的男女、支持不同平台和不同文件格式的媒体播放器等。如果用桥接模式就能很好地解决这些问题。
桥接(Bridge)模式的定义如下:将抽象与实现分离,使它们可以独立变化。它是用组合关系代替继承关系来实现,从而降低了抽象和实现这两个可变维度的耦合度。
通过上面的讲解,我们能很好的感觉到桥接模式遵循了里氏替换原则和依赖倒置原则,最终实现了开闭原则,对修改关闭,对扩展开放。这里将桥接模式的优缺点总结如下。
桥接(Bridge)模式的优点是:
缺点是:由于聚合关系建立在抽象层,要求开发者针对抽象化进行设计与编程,能正确地识别出系统中两个独立变化的维度,这增加了系统的理解与设计难度。
可以将抽象化部分与实现化部分分开,取消二者的继承关系,改用组合关系。
1. 模式的结构
桥接(Bridge)模式包含以下主要角色。
其结构图如图 1 所示。
图1 桥接模式的结构图
2. 模式的实现
桥接模式的代码如下:
- package bridge;
- public class BridgeTest {
- public static void main(String[] args) {
- Implementor imple = new ConcreteImplementorA();
- Abstraction abs = new RefinedAbstraction(imple);
- abs.Operation();
- }
- }
- //实现化角色
- interface Implementor {
- public void OperationImpl();
- }
- //具体实现化角色
- class ConcreteImplementorA implements Implementor {
- public void OperationImpl() {
- System.out.println("具体实现化(Concrete Implementor)角色被访问");
- }
- }
- //抽象化角色
- abstract class Abstraction {
- protected Implementor imple;
- protected Abstraction(Implementor imple) {
- this.imple = imple;
- }
- public abstract void Operation();
- }
- //扩展抽象化角色
- class RefinedAbstraction extends Abstraction {
- protected RefinedAbstraction(Implementor imple) {
- super(imple);
- }
- public void Operation() {
- System.out.println("扩展抽象化(Refined Abstraction)角色被访问");
- imple.OperationImpl();
- }
- }
程序的运行结果如下:
扩展抽象化(Refined Abstraction)角色被访问 具体实现化(Concrete Implementor)角色被访问
【例1】用桥接(Bridge)模式模拟女士皮包的选购。
分析:女士皮包有很多种,可以按用途分、按皮质分、按品牌分、按颜色分、按大小分等,存在多个维度的变化,所以采用桥接模式来实现女士皮包的选购比较合适。
本实例按用途分可选钱包(Wallet)和挎包(HandBag),按颜色分可选黄色(Yellow)和红色(Red)。可以按两个维度定义为颜色类和包类。(点此下载本实例所要显示的包的图片)。
颜色类(Color)是一个维度,定义为实现化角色,它有两个具体实现化角色:黄色和红色,通过 getColor() 方法可以选择颜色;包类(Bag)是另一个维度,定义为抽象化角色,它有两个扩展抽象化角色:挎包和钱包,它包含了颜色类对象,通过 getName() 方法可以选择相关颜色的挎包和钱包。
客户类通过 ReadXML 类从 XML 配置文件中获取包信息(点此下载 XML 配置文件),并把选到的产品通过窗体显示出现,图 2 所示是其结构图。
图2 女士皮包选购的结构图
程序代码如下:
- package bridge;
- import org.w3c.dom.NodeList;
- import javax.swing.*;
- import javax.xml.parsers.DocumentBuilder;
- import javax.xml.parsers.DocumentBuilderFactory;
- import java.awt.*;
- public class BagManage {
- public static void main(String[] args) {
- Color color;
- Bag bag;
- color = (Color) ReadXML.getObject("color");
- bag = (Bag) ReadXML.getObject("bag");
- bag.setColor(color);
- String name = bag.getName();
- show(name);
- }
- public static void show(String name) {
- JFrame jf = new JFrame("桥接模式测试");
- Container contentPane = jf.getContentPane();
- JPanel p = new JPanel();
- JLabel l = new JLabel(new ImageIcon("src/bridge/" + name + ".jpg"));
- p.setLayout(new GridLayout(1, 1));
- p.setBorder(BorderFactory.createTitledBorder("女士皮包"));
- p.add(l);
- contentPane.add(p, BorderLayout.CENTER);
- jf.pack();
- jf.setVisible(true);
- jf.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
- }
- }
- //实现化角色:颜色
- interface Color {
- String getColor();
- }
- //具体实现化角色:黄色
- class Yellow implements Color {
- public String getColor() {
- return "yellow";
- }
- }
- //具体实现化角色:红色
- class Red implements Color {
- public String getColor() {
- return "red";
- }
- }
- //抽象化角色:包
- abstract class Bag {
- protected Color color;
- public void setColor(Color color) {
- this.color = color;
- }
- public abstract String getName();
- }
- //扩展抽象化角色:挎包
- class HandBag extends Bag {
- public String getName() {
- return color.getColor() + "HandBag";
- }
- }
- //扩展抽象化角色:钱包
- class Wallet extends Bag {
- public String getName() {
- return color.getColor() + "Wallet";
- }
- }
- package bridge;
- import javax.xml.parsers.*;
- import org.w3c.dom.*;
- import java.io.*;
- class ReadXML {
- public static Object getObject(String args) {
- try {
- DocumentBuilderFactory dFactory = DocumentBuilderFactory.newInstance();
- DocumentBuilder builder = dFactory.newDocumentBuilder();
- Document doc;
- doc = builder.parse(new File("src/bridge/config.xml"));
- NodeList nl = doc.getElementsByTagName("className");
- Node classNode = null;
- if (args.equals("color")) {
- classNode = nl.item(0).getFirstChild();
- } else if (args.equals("bag")) {
- classNode = nl.item(1).getFirstChild();
- }
- String cName = "bridge." + classNode.getNodeValue();
- Class<?> c = Class.forName(cName);
- Object obj = c.newInstance();
- return obj;
- } catch (Exception e) {
- e.printStackTrace();
- return null;
- }
- }
- }
程序的运行结果如图 3 所示。
图3 女士皮包选购的运行结果1
如果将 XML 配置文件按如下修改:
- <?xml version="1.0" encoding="UTF-8"?>
- <config>
- <className>Red</className>
- <className>Wallet</className>
- </config>
则程序的运行结果如图 4 所示。
图4 女士皮包选购的运行结果2
当一个类内部具备两种或多种变化维度时,使用桥接模式可以解耦这些变化的维度,使高层代码架构稳定。
桥接模式通常适用于以下场景。
桥接模式的一个常见使用场景就是替换继承。我们知道,继承拥有很多优点,比如,抽象、封装、多态等,父类封装共性,子类实现特性。继承可以很好的实现代码复用(封装)的功能,但这也是继承的一大缺点。
因为父类拥有的方法,子类也会继承得到,无论子类需不需要,这说明继承具备强侵入性(父类代码侵入子类),同时会导致子类臃肿。因此,在设计模式中,有一个原则为优先使用组合/聚合,而不是继承。
很多时候,我们分不清该使用继承还是组合/聚合或其他方式等,其实可以从现实语义进行思考。因为软件最终还是提供给现实生活中的人使用的,是服务于人类社会的,软件是具备现实场景的。当我们从纯代码角度无法看清问题时,现实角度可能会提供更加开阔的思路。
在软件开发中,有时桥接(Bridge)模式可与适配器模式联合使用。当桥接(Bridge)模式的实现化角色的接口与现有类的接口不一致时,可以在二者中间定义一个适配器将二者连接起来,其具体结构图如图 5 所示。
图5 桥接模式与适配器模式联用的结构图
上班族大多都有睡懒觉的习惯,每天早上上班时间都很紧张,于是很多人为了多睡一会,就会用方便的方式解决早餐问题。有些人早餐可能会吃煎饼,煎饼中可以加鸡蛋,也可以加香肠,但是不管怎么“加码”,都还是一个煎饼。在现实生活中,常常需要对现有产品增加新的功能或美化其外观,如房子装修、相片加相框等,都是装饰器模式。
在软件开发过程中,有时想用一些现存的组件。这些组件可能只是完成了一些核心功能。但在不改变其结构的情况下,可以动态地扩展其功能。所有这些都可以釆用装饰器模式来实现。
装饰器(Decorator)模式的定义:指在不改变现有对象结构的情况下,动态地给该对象增加一些职责(即增加其额外功能)的模式,它属于对象结构型模式。
装饰器模式的主要优点有:
其主要缺点是:装饰器模式会增加许多子类,过度使用会增加程序得复杂性。
通常情况下,扩展一个类的功能会使用继承方式来实现。但继承具有静态特征,耦合度高,并且随着扩展功能的增多,子类会很膨胀。如果使用组合关系来创建一个包装对象(即装饰对象)来包裹真实对象,并在保持真实对象的类结构不变的前提下,为其提供额外的功能,这就是装饰器模式的目标。下面来分析其基本结构和实现方法。
1. 模式的结构
装饰器模式主要包含以下角色。
装饰器模式的结构图如图 1 所示。
图1 装饰器模式的结构图
2. 模式的实现
装饰器模式的实现代码如下:
- package decorator;
- public class DecoratorPattern {
- public static void main(String[] args) {
- Component p = new ConcreteComponent();
- p.operation();
- System.out.println("---------------------------------");
- Component d = new ConcreteDecorator(p);
- d.operation();
- }
- }
- //抽象构件角色
- interface Component {
- public void operation();
- }
- //具体构件角色
- class ConcreteComponent implements Component {
- public ConcreteComponent() {
- System.out.println("创建具体构件角色");
- }
- public void operation() {
- System.out.println("调用具体构件角色的方法operation()");
- }
- }
- //抽象装饰角色
- class Decorator implements Component {
- private Component component;
- public Decorator(Component component) {
- this.component = component;
- }
- public void operation() {
- component.operation();
- }
- }
- //具体装饰角色
- class ConcreteDecorator extends Decorator {
- public ConcreteDecorator(Component component) {
- super(component);
- }
- public void operation() {
- super.operation();
- addedFunction();
- }
- public void addedFunction() {
- System.out.println("为具体构件角色增加额外的功能addedFunction()");
- }
- }
程序运行结果如下:
创建具体构件角色 调用具体构件角色的方法operation() --------------------------------- 调用具体构件角色的方法operation() 为具体构件角色增加额外的功能addedFunction()
【例1】用装饰器模式实现游戏角色“莫莉卡·安斯兰”的变身。
分析:在《恶魔战士》中,游戏角色“莫莉卡·安斯兰”的原身是一个可爱少女,但当她变身时,会变成头顶及背部延伸出蝙蝠状飞翼的女妖,当然她还可以变为穿着漂亮外衣的少女。这些都可用装饰器模式来实现,在本实例中的“莫莉卡”原身有 setImage(String t) 方法决定其显示方式,而其 变身“蝙蝠状女妖”和“着装少女”可以用 setChanger() 方法来改变其外观,原身与变身后的效果用 display() 方法来显示(点此下载其原身和变身后的图片),图 2 所示是其结构图。
图2 游戏角色“莫莉卡·安斯兰”的结构图
程序代码如下:
- package decorator;
- import java.awt.*;
- import javax.swing.*;
- public class MorriganAensland {
- public static void main(String[] args) {
- Morrigan m0 = new original();
- m0.display();
- Morrigan m1 = new Succubus(m0);
- m1.display();
- Morrigan m2 = new Girl(m0);
- m2.display();
- }
- }
- //抽象构件角色:莫莉卡
- interface Morrigan {
- public void display();
- }
- //具体构件角色:原身
- class original extends JFrame implements Morrigan {
- private static final long serialVersionUID = 1L;
- private String t = "Morrigan0.jpg";
- public original() {
- super("《恶魔战士》中的莫莉卡·安斯兰");
- }
- public void setImage(String t) {
- this.t = t;
- }
- public void display() {
- this.setLayout(new FlowLayout());
- JLabel l1 = new JLabel(new ImageIcon("src/decorator/" + t));
- this.add(l1);
- this.pack();
- this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
- this.setVisible(true);
- }
- }
- //抽象装饰角色:变形
- class Changer implements Morrigan {
- Morrigan m;
- public Changer(Morrigan m) {
- this.m = m;
- }
- public void display() {
- m.display();
- }
- }
- //具体装饰角色:女妖
- class Succubus extends Changer {
- public Succubus(Morrigan m) {
- super(m);
- }
- public void display() {
- setChanger();
- super.display();
- }
- public void setChanger() {
- ((original) super.m).setImage("Morrigan1.jpg");
- }
- }
- //具体装饰角色:少女
- class Girl extends Changer {
- public Girl(Morrigan m) {
- super(m);
- }
- public void display() {
- setChanger();
- super.display();
- }
- public void setChanger() {
- ((original) super.m).setImage("Morrigan2.jpg");
- }
- }
程序运行结果如图 3 所示。
图3 游戏角色“莫莉卡·安斯兰”的变身
前面讲解了关于装饰器模式的结构与特点,下面介绍其适用的应用场景,装饰器模式通常在以下几种情况使用。
装饰器模式在 Java 语言中的最著名的应用莫过于 Java I/O 标准库的设计了。例如,InputStream 的子类 FilterInputStream,OutputStream 的子类 FilterOutputStream,Reader 的子类 BufferedReader 以及 FilterReader,还有 Writer 的子类 BufferedWriter、FilterWriter 以及 PrintWriter 等,它们都是抽象装饰类。
下面代码是为 FileReader 增加缓冲区而采用的装饰类 BufferedReader 的例子:
- BufferedReader in = new BufferedReader(new FileReader("filename.txt"));
- String s = in.readLine();
装饰器模式所包含的 4 个角色不是任何时候都要存在的,在有些应用环境下模式是可以简化的,如以下两种情况。
(1) 如果只有一个具体构件而没有抽象构件时,可以让抽象装饰继承具体构件,其结构图如图 4 所示。
图4 只有一个具体构件的装饰器模式
(2) 如果只有一个具体装饰时,可以将抽象装饰和具体装饰合并,其结构图如图 5 所示。
图5 只有一个具体装饰的装饰器模式
在现实生活中,常常存在办事较复杂的例子,如办房产证或注册一家公司,有时要同多个部门联系,这时要是有一个综合部门能解决一切手续问题就好了。
软件设计也是这样,当一个系统的功能越来越强,子系统会越来越多,客户对系统的访问也变得越来越复杂。这时如果系统内部发生改变,客户端也要跟着改变,这违背了“开闭原则”,也违背了“迪米特法则”,所以有必要为多个子系统提供一个统一的接口,从而降低系统的耦合度,这就是外观模式的目标。
图 1 给出了客户去当地房产局办理房产证过户要遇到的相关部门。
图1 办理房产证过户的相关部门
外观(Facade)模式又叫作门面模式,是一种通过为多个复杂的子系统提供一个一致的接口,而使这些子系统更加容易被访问的模式。该模式对外有一个统一接口,外部应用程序不用关心内部子系统的具体细节,这样会大大降低应用程序的复杂度,提高了程序的可维护性。
在日常编码工作中,我们都在有意无意的大量使用外观模式。只要是高层模块需要调度多个子系统(2个以上的类对象),我们都会自觉地创建一个新的类封装这些子系统,提供精简的接口,让高层模块可以更加容易地间接调用这些子系统的功能。尤其是现阶段各种第三方SDK、开源类库,很大概率都会使用外观模式。
外观(Facade)模式是“迪米特法则”的典型应用,它有以下主要优点。
外观(Facade)模式的主要缺点如下。
外观(Facade)模式的结构比较简单,主要是定义了一个高层接口。它包含了对各个子系统的引用,客户端可以通过它访问各个子系统的功能。现在来分析其基本结构和实现方法。
1. 模式的结构
外观(Facade)模式包含以下主要角色。
其结构图如图 2 所示。
图2 外观(Facade)模式的结构图
2. 模式的实现
外观模式的实现代码如下:
- package facade;
- public class FacadePattern {
- public static void main(String[] args) {
- Facade f = new Facade();
- f.method();
- }
- }
- //外观角色
- class Facade {
- private SubSystem01 obj1 = new SubSystem01();
- private SubSystem02 obj2 = new SubSystem02();
- private SubSystem03 obj3 = new SubSystem03();
- public void method() {
- obj1.method1();
- obj2.method2();
- obj3.method3();
- }
- }
- //子系统角色
- class SubSystem01 {
- public void method1() {
- System.out.println("子系统01的method1()被调用!");
- }
- }
- //子系统角色
- class SubSystem02 {
- public void method2() {
- System.out.println("子系统02的method2()被调用!");
- }
- }
- //子系统角色
- class SubSystem03 {
- public void method3() {
- System.out.println("子系统03的method3()被调用!");
- }
- }
程序运行结果如下:
子系统01的method1()被调用! 子系统02的method2()被调用! 子系统03的method3()被调用!
【例1】用“外观模式”设计一个婺源特产的选购界面。
分析:本实例的外观角色 WySpecialty 是 JPanel 的子类,它拥有 8 个子系统角色 Specialty1~Specialty8,它们是图标类(ImageIcon)的子类对象,用来保存该婺源特产的图标(点此下载要显示的婺源特产的图片)。
外观类(WySpecialty)用 JTree 组件来管理婺源特产的名称,并定义一个事件处理方法 valueClianged(TreeSelectionEvent e),当用户从树中选择特产时,该特产的图标对象保存在标签(JLabd)对象中。
客户窗体对象用分割面板来实现,左边放外观角色的目录树,右边放显示所选特产图像的标签。其结构图如图 3 所示。
图3 婺源特产管理界面的结构图
程序代码如下:
- package facade;
- import java.awt.*;
- import javax.swing.*;
- import javax.swing.event.*;
- import javax.swing.tree.DefaultMutableTreeNode;
- public class WySpecialtyFacade {
- public static void main(String[] args) {
- JFrame f = new JFrame("外观模式: 婺源特产选择测试");
- Container cp = f.getContentPane();
- WySpecialty wys = new WySpecialty();
- JScrollPane treeView = new JScrollPane(wys.tree);
- JScrollPane scrollpane = new JScrollPane(wys.label);
- JSplitPane splitpane = new JSplitPane(JSplitPane.HORIZONTAL_SPLIT, true, treeView, scrollpane); //分割面版
- splitpane.setDividerLocation(230); //设置splitpane的分隔线位置
- splitpane.setOneTouchExpandable(true); //设置splitpane可以展开或收起
- cp.add(splitpane);
- f.setSize(650, 350);
- f.setVisible(true);
- f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
- }
- }
- class WySpecialty extends JPanel implements TreeSelectionListener {
- private static final long serialVersionUID = 1L;
- final JTree tree;
- JLabel label;
- private Specialty1 s1 = new Specialty1();
- private Specialty2 s2 = new Specialty2();
- private Specialty3 s3 = new Specialty3();
- private Specialty4 s4 = new Specialty4();
- private Specialty5 s5 = new Specialty5();
- private Specialty6 s6 = new Specialty6();
- private Specialty7 s7 = new Specialty7();
- private Specialty8 s8 = new Specialty8();
- WySpecialty() {
- DefaultMutableTreeNode top = new DefaultMutableTreeNode("婺源特产");
- DefaultMutableTreeNode node1 = null, node2 = null, tempNode = null;
- node1 = new DefaultMutableTreeNode("婺源四大特产(红、绿、黑、白)");
- tempNode = new DefaultMutableTreeNode("婺源荷包红鲤鱼");
- node1.add(tempNode);
- tempNode = new DefaultMutableTreeNode("婺源绿茶");
- node1.add(tempNode);
- tempNode = new DefaultMutableTreeNode("婺源龙尾砚");
- node1.add(tempNode);
- tempNode = new DefaultMutableTreeNode("婺源江湾雪梨");
- node1.add(tempNode);
- top.add(node1);
- node2 = new DefaultMutableTreeNode("婺源其它土特产");
- tempNode = new DefaultMutableTreeNode("婺源酒糟鱼");
- node2.add(tempNode);
- tempNode = new DefaultMutableTreeNode("婺源糟米子糕");
- node2.add(tempNode);
- tempNode = new DefaultMutableTreeNode("婺源清明果");
- node2.add(tempNode);
- tempNode = new DefaultMutableTreeNode("婺源油煎灯");
- node2.add(tempNode);
- top.add(node2);
- tree = new JTree(top);
- tree.addTreeSelectionListener(this);
- label = new JLabel();
- }
- public void valueChanged(TreeSelectionEvent e) {
- if (e.getSource() == tree) {
- DefaultMutableTreeNode node = (DefaultMutableTreeNode) tree.getLastSelectedPathComponent();
- if (node == null) return;
- if (node.isLeaf()) {
- Object object = node.getUserObject();
- String sele = object.toString();
- label.setText(sele);
- label.setHorizontalTextPosition(JLabel.CENTER);
- label.setVerticalTextPosition(JLabel.BOTTOM);
- sele = sele.substring(2, 4);
- if (sele.equalsIgnoreCase("荷包")) label.setIcon(s1);
- else if (sele.equalsIgnoreCase("绿茶")) label.setIcon(s2);
- else if (sele.equalsIgnoreCase("龙尾")) label.setIcon(s3);
- else if (sele.equalsIgnoreCase("江湾")) label.setIcon(s4);
- else if (sele.equalsIgnoreCase("酒糟")) label.setIcon(s5);
- else if (sele.equalsIgnoreCase("糟米")) label.setIcon(s6);
- else if (sele.equalsIgnoreCase("清明")) label.setIcon(s7);
- else if (sele.equalsIgnoreCase("油煎")) label.setIcon(s8);
- label.setHorizontalAlignment(JLabel.CENTER);
- }
- }
- }
- }
- class Specialty1 extends ImageIcon {
- private static final long serialVersionUID = 1L;
- Specialty1() {
- super("src/facade/WyImage/Specialty11.jpg");
- }
- }
- class Specialty2 extends ImageIcon {
- private static final long serialVersionUID = 1L;
- Specialty2() {
- super("src/facade/WyImage/Specialty12.jpg");
- }
- }
- class Specialty3 extends ImageIcon {
- private static final long serialVersionUID = 1L;
- Specialty3() {
- super("src/facade/WyImage/Specialty13.jpg");
- }
- }
- class Specialty4 extends ImageIcon {
- private static final long serialVersionUID = 1L;
- Specialty4() {
- super("src/facade/WyImage/Specialty14.jpg");
- }
- }
- class Specialty5 extends ImageIcon {
- private static final long serialVersionUID = 1L;
- Specialty5() {
- super("src/facade/WyImage/Specialty21.jpg");
- }
- }
- class Specialty6 extends ImageIcon {
- private static final long serialVersionUID = 1L;
- Specialty6() {
- super("src/facade/WyImage/Specialty22.jpg");
- }
- }
- class Specialty7 extends ImageIcon {
- private static final long serialVersionUID = 1L;
- Specialty7() {
- super("src/facade/WyImage/Specialty23.jpg");
- }
- }
- class Specialty8 extends ImageIcon {
- private static final long serialVersionUID = 1L;
- Specialty8() {
- super("src/facade/WyImage/Specialty24.jpg");
- }
- }
程序运行结果如图 4 所示。
图4 婺源特产管理界面的运行结果
通常在以下情况下可以考虑使用外观模式。
在外观模式中,当增加或移除子系统时需要修改外观类,这违背了“开闭原则”。如果引入抽象外观类,则在一定程度上解决了该问题,其结构图如图 5 所示。
图5 引入抽象外观类的外观模式的结构图
在面向对象程序设计过程中,有时会面临要创建大量相同或相似对象实例的问题。创建那么多的对象将会耗费很多的系统资源,它是系统性能提高的一个瓶颈。
例如,围棋和五子棋中的黑白棋子,图像中的坐标点或颜色,局域网中的路由器、交换机和集线器,教室里的桌子和凳子等。这些对象有很多相似的地方,如果能把它们相同的部分提取出来共享,则能节省大量的系统资源,这就是享元模式的产生背景。
享元(Flyweight)模式的定义:运用共享技术来有效地支持大量细粒度对象的复用。它通过共享已经存在的对象来大幅度减少需要创建的对象数量、避免大量相似类的开销,从而提高系统资源的利用率。
享元模式的主要优点是:相同对象只要保存一份,这降低了系统中对象的数量,从而降低了系统中细粒度对象给内存带来的压力。
其主要缺点是:
享元模式的定义提出了两个要求,细粒度和共享对象。因为要求细粒度,所以不可避免地会使对象数量多且性质相近,此时我们就将这些对象的信息分为两个部分:内部状态和外部状态。
比如,连接池中的连接对象,保存在连接对象中的用户名、密码、连接URL等信息,在创建对象的时候就设置好了,不会随环境的改变而改变,这些为内部状态。而当每个连接要被回收利用时,我们需要将它标记为可用状态,这些为外部状态。
享元模式的本质是缓存共享对象,降低内存消耗。
1. 模式的结构
享元模式的主要角色有如下。
图 1 是享元模式的结构图,其中:
图1 享元模式的结构图
2. 模式的实现
享元模式的实现代码如下:
- public class FlyweightPattern {
- public static void main(String[] args) {
- FlyweightFactory factory = new FlyweightFactory();
- Flyweight f01 = factory.getFlyweight("a");
- Flyweight f02 = factory.getFlyweight("a");
- Flyweight f03 = factory.getFlyweight("a");
- Flyweight f11 = factory.getFlyweight("b");
- Flyweight f12 = factory.getFlyweight("b");
- f01.operation(new UnsharedConcreteFlyweight("第1次调用a。"));
- f02.operation(new UnsharedConcreteFlyweight("第2次调用a。"));
- f03.operation(new UnsharedConcreteFlyweight("第3次调用a。"));
- f11.operation(new UnsharedConcreteFlyweight("第1次调用b。"));
- f12.operation(new UnsharedConcreteFlyweight("第2次调用b。"));
- }
- }
- //非享元角色
- class UnsharedConcreteFlyweight {
- private String info;
- UnsharedConcreteFlyweight(String info) {
- this.info = info;
- }
- public String getInfo() {
- return info;
- }
- public void setInfo(String info) {
- this.info = info;
- }
- }
- //抽象享元角色
- interface Flyweight {
- public void operation(UnsharedConcreteFlyweight state);
- }
- //具体享元角色
- class ConcreteFlyweight implements Flyweight {
- private String key;
- ConcreteFlyweight(String key) {
- this.key = key;
- System.out.println("具体享元" + key + "被创建!");
- }
- public void operation(UnsharedConcreteFlyweight outState) {
- System.out.print("具体享元" + key + "被调用,");
- System.out.println("非享元信息是:" + outState.getInfo());
- }
- }
- //享元工厂角色
- class FlyweightFactory {
- private HashMap<String, Flyweight> flyweights = new HashMap<String, Flyweight>();
- public Flyweight getFlyweight(String key) {
- Flyweight flyweight = (Flyweight) flyweights.get(key);
- if (flyweight != null) {
- System.out.println("具体享元" + key + "已经存在,被成功获取!");
- } else {
- flyweight = new ConcreteFlyweight(key);
- flyweights.put(key, flyweight);
- }
- return flyweight;
- }
- }
程序运行结果如下:
具体享元a被创建! 具体享元a已经存在,被成功获取! 具体享元a已经存在,被成功获取! 具体享元b被创建! 具体享元b已经存在,被成功获取! 具体享元a被调用,非享元信息是:第1次调用a。 具体享元a被调用,非享元信息是:第2次调用a。 具体享元a被调用,非享元信息是:第3次调用a。 具体享元b被调用,非享元信息是:第1次调用b。 具体享元b被调用,非享元信息是:第2次调用b。
【例1】享元模式在五子棋游戏中的应用。
分析:五子棋同围棋一样,包含多个“黑”或“白”颜色的棋子,所以用享元模式比较好。
本实例中:
图 2 所示是其结构图。
图2 五子棋游戏的结构图
程序代码如下:
- import javax.swing.*;
- import java.awt.*;
- import java.awt.event.MouseAdapter;
- import java.awt.event.MouseEvent;
- import java.util.ArrayList;
- public class WzqGame {
- public static void main(String[] args) {
- new Chessboard();
- }
- }
- //棋盘
- class Chessboard extends MouseAdapter {
- WeiqiFactory wf;
- JFrame f;
- Graphics g;
- JRadioButton wz;
- JRadioButton bz;
- private final int x = 50;
- private final int y = 50;
- private final int w = 40; //小方格宽度和高度
- private final int rw = 400; //棋盘宽度和高度
- Chessboard() {
- wf = new WeiqiFactory();
- f = new JFrame("享元模式在五子棋游戏中的应用");
- f.setBounds(100, 100, 500, 550);
- f.setVisible(true);
- f.setResizable(false);
- f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
- JPanel SouthJP = new JPanel();
- f.add("South", SouthJP);
- wz = new JRadioButton("白子");
- bz = new JRadioButton("黑子", true);
- ButtonGroup group = new ButtonGroup();
- group.add(wz);
- group.add(bz);
- SouthJP.add(wz);
- SouthJP.add(bz);
- JPanel CenterJP = new JPanel();
- CenterJP.setLayout(null);
- CenterJP.setSize(500, 500);
- CenterJP.addMouseListener(this);
- f.add("Center", CenterJP);
- try {
- Thread.sleep(500);
- } catch (InterruptedException e) {
- e.printStackTrace();
- }
- g = CenterJP.getGraphics();
- g.setColor(Color.BLUE);
- g.drawRect(x, y, rw, rw);
- for (int i = 1; i < 10; i++) {
- //绘制第i条竖直线
- g.drawLine(x + (i * w), y, x + (i * w), y + rw);
- //绘制第i条水平线
- g.drawLine(x, y + (i * w), x + rw, y + (i * w));
- }
- }
- public void mouseClicked(MouseEvent e) {
- Point pt = new Point(e.getX() - 15, e.getY() - 15);
- if (wz.isSelected()) {
- ChessPieces c1 = wf.getChessPieces("w");
- c1.DownPieces(g, pt);
- } else if (bz.isSelected()) {
- ChessPieces c2 = wf.getChessPieces("b");
- c2.DownPieces(g, pt);
- }
- }
- }
- //抽象享元角色:棋子
- interface ChessPieces {
- public void DownPieces(Graphics g, Point pt); //下子
- }
- //具体享元角色:白子
- class WhitePieces implements ChessPieces {
- public void DownPieces(Graphics g, Point pt) {
- g.setColor(Color.WHITE);
- g.fillOval(pt.x, pt.y, 30, 30);
- }
- }
- //具体享元角色:黑子
- class BlackPieces implements ChessPieces {
- public void DownPieces(Graphics g, Point pt) {
- g.setColor(Color.BLACK);
- g.fillOval(pt.x, pt.y, 30, 30);
- }
- }
- //享元工厂角色
- class WeiqiFactory {
- private ArrayList<ChessPieces> qz;
- public WeiqiFactory() {
- qz = new ArrayList<ChessPieces>();
- ChessPieces w = new WhitePieces();
- qz.add(w);
- ChessPieces b = new BlackPieces();
- qz.add(b);
- }
- public ChessPieces getChessPieces(String type) {
- if (type.equalsIgnoreCase("w")) {
- return (ChessPieces) qz.get(0);
- } else if (type.equalsIgnoreCase("b")) {
- return (ChessPieces) qz.get(1);
- } else {
- return null;
- }
- }
- }
程序运行结果如图 3 所示。
图3 五子棋游戏的运行结果
当系统中多处需要同一组信息时,可以把这些信息封装到一个对象中,然后对该对象进行缓存,这样,一个对象就可以提供给多出需要使用的地方,避免大量同一对象的多次创建,降低大量内存空间的消耗。
享元模式其实是工厂方法模式的一个改进机制,享元模式同样要求创建一个或一组对象,并且就是通过工厂方法模式生成对象的,只不过享元模式为工厂方法模式增加了缓存这一功能。
前面分析了享元模式的结构与特点,下面分析它适用的应用场景。享元模式是通过减少内存中对象的数量来节省内存空间的,所以以下几种情形适合采用享元模式。
在前面介绍的享元模式中,其结构图通常包含可以共享的部分和不可以共享的部分。在实际使用过程中,有时候会稍加改变,即存在两种特殊的享元模式:单纯享元模式和复合享元模式,下面分别对它们进行简单介绍。
(1) 单纯享元模式,这种享元模式中的所有的具体享元类都是可以共享的,不存在非共享的具体享元类,其结构图如图 4 所示。
图4 单纯享元模式的结构图
(2) 复合享元模式,这种享元模式中的有些享元对象是由一些单纯享元对象组合而成的,它们就是复合享元对象。虽然复合享元对象本身不能共享,但它们可以分解成单纯享元对象再被共享,其结构图如图 5 所示。
图5 复合享元模式的结构图
在现实生活中,存在很多“部分-整体”的关系,例如,大学中的部门与学院、总公司中的部门与分公司、学习用品中的书与书包、生活用品中的衣服与衣柜、以及厨房中的锅碗瓢盆等。在软件开发中也是这样,例如,文件系统中的文件与文件夹、窗体程序中的简单控件与容器控件等。对这些简单对象与复合对象的处理,如果用组合模式来实现会很方便。
组合(Composite Pattern)模式的定义:有时又叫作整体-部分(Part-Whole)模式,它是一种将对象组合成树状的层次结构的模式,用来表示“整体-部分”的关系,使用户对单个对象和组合对象具有一致的访问性,属于结构型设计模式。
组合模式一般用来描述整体与部分的关系,它将对象组织到树形结构中,顶层的节点被称为根节点,根节点下面可以包含树枝节点和叶子节点,树枝节点下面又可以包含树枝节点和叶子节点,树形结构图如下。
由上图可以看出,其实根节点和树枝节点本质上属于同一种数据类型,可以作为容器使用;而叶子节点与树枝节点在语义上不属于用一种类型。但是在组合模式中,会把树枝节点和叶子节点看作属于同一种数据类型(用统一接口定义),让它们具备一致行为。
这样,在组合模式中,整个树形结构中的对象都属于同一种类型,带来的好处就是用户不需要辨别是树枝节点还是叶子节点,可以直接进行操作,给用户的使用带来极大的便利。
组合模式的主要优点有:
其主要缺点是:
组合模式的结构不是很复杂,下面对它的结构和实现进行分析。
组合模式包含以下主要角色。
组合模式分为透明式的组合模式和安全式的组合模式。
(1) 透明方式
在该方式中,由于抽象构件声明了所有子类中的全部方法,所以客户端无须区别树叶对象和树枝对象,对客户端来说是透明的。但其缺点是:树叶构件本来没有 Add()、Remove() 及 GetChild() 方法,却要实现它们(空实现或抛异常),这样会带来一些安全性问题。其结构图如图 1 所示。
图1 透明式的组合模式的结构图
(2) 安全方式
在该方式中,将管理子构件的方法移到树枝构件中,抽象构件和树叶构件没有对子对象的管理方法,这样就避免了上一种方式的安全性问题,但由于叶子和分支有不同的接口,客户端在调用时要知道树叶对象和树枝对象的存在,所以失去了透明性。其结构图如图 2 所示。
图2 安全式的组合模式的结构图
假如要访问集合 c0={leaf1,{leaf2,leaf3}} 中的元素,其对应的树状图如图 3 所示。
图3 集合c0的树状图
透明组合模式
下面为透明式的组合模式的实现代码。
- public class CompositePattern {
- public static void main(String[] args) {
- Component c0 = new Composite();
- Component c1 = new Composite();
- Component leaf1 = new Leaf("1");
- Component leaf2 = new Leaf("2");
- Component leaf3 = new Leaf("3");
- c0.add(leaf1);
- c0.add(c1);
- c1.add(leaf2);
- c1.add(leaf3);
- c0.operation();
- }
- }
- //抽象构件
- interface Component {
- public void add(Component c);
- public void remove(Component c);
- public Component getChild(int i);
- public void operation();
- }
- //树叶构件
- class Leaf implements Component {
- private String name;
- public Leaf(String name) {
- this.name = name;
- }
- public void add(Component c) {
- }
- public void remove(Component c) {
- }
- public Component getChild(int i) {
- return null;
- }
- public void operation() {
- System.out.println("树叶" + name + ":被访问!");
- }
- }
- //树枝构件
- class Composite implements Component {
- private ArrayList<Component> children = new ArrayList<Component>();
- public void add(Component c) {
- children.add(c);
- }
- public void remove(Component c) {
- children.remove(c);
- }
- public Component getChild(int i) {
- return children.get(i);
- }
- public void operation() {
- for (Object obj : children) {
- ((Component) obj).operation();
- }
- }
- }
程序运行结果如下:
树叶1:被访问! 树叶2:被访问! 树叶3:被访问!
安全组合模式
安全式的组合模式与透明式组合模式的实现代码类似,只要对其做简单修改就可以了,代码如下。
首先修改 Component 代码,只保留层次的公共行为。
- interface Component {
- public void operation();
- }
然后修改客户端代码,将树枝构件类型更改为 Composite 类型,以便获取管理子类操作的方法。
- public class CompositePattern {
- public static void main(String[] args) {
- Composite c0 = new Composite();
- Composite c1 = new Composite();
- Component leaf1 = new Leaf("1");
- Component leaf2 = new Leaf("2");
- Component leaf3 = new Leaf("3");
- c0.add(leaf1);
- c0.add(c1);
- c1.add(leaf2);
- c1.add(leaf3);
- c0.operation();
- }
- }
【例1】用组合模式实现当用户在商店购物后,显示其所选商品信息,并计算所选商品总价的功能。
说明:假如李先生到韶关“天街e角”生活用品店购物,用 1 个红色小袋子装了 2 包婺源特产(单价 7.9 元)、1 张婺源地图(单价 9.9 元);用 1 个白色小袋子装了 2 包韶关香藉(单价 68 元)和 3 包韶关红茶(单价 180 元);用 1 个中袋子装了前面的红色小袋子和 1 个景德镇瓷器(单价 380 元);用 1 个大袋子装了前面的中袋子、白色小袋子和 1 双李宁牌运动鞋(单价 198 元)。
最后“大袋子”中的内容有:{1 双李宁牌运动鞋(单价 198 元)、白色小袋子{2 包韶关香菇(单价 68 元)、3 包韶关红茶(单价 180 元)}、中袋子{1 个景德镇瓷器(单价 380 元)、红色小袋子{2 包婺源特产(单价 7.9 元)、1 张婺源地图(单价 9.9 元)}}},现在要求编程显示李先生放在大袋子中的所有商品信息并计算要支付的总价。
本实例可按安全组合模式设计,其结构图如图 4 所示。
图4 韶关“天街e角”店购物的结构图
程序代码如下:
- package composite;
- import java.util.ArrayList;
- public class ShoppingTest {
- public static void main(String[] args) {
- float s = 0;
- Bags BigBag, mediumBag, smallRedBag, smallWhiteBag;
- Goods sp;
- BigBag = new Bags("大袋子");
- mediumBag = new Bags("中袋子");
- smallRedBag = new Bags("红色小袋子");
- smallWhiteBag = new Bags("白色小袋子");
- sp = new Goods("婺源特产", 2, 7.9f);
- smallRedBag.add(sp);
- sp = new Goods("婺源地图", 1, 9.9f);
- smallRedBag.add(sp);
- sp = new Goods("韶关香菇", 2, 68);
- smallWhiteBag.add(sp);
- sp = new Goods("韶关红茶", 3, 180);
- smallWhiteBag.add(sp);
- sp = new Goods("景德镇瓷器", 1, 380);
- mediumBag.add(sp);
- mediumBag.add(smallRedBag);
- sp = new Goods("李宁牌运动鞋", 1, 198);
- BigBag.add(sp);
- BigBag.add(smallWhiteBag);
- BigBag.add(mediumBag);
- System.out.println("您选购的商品有:");
- BigBag.show();
- s = BigBag.calculation();
- System.out.println("要支付的总价是:" + s + "元");
- }
- }
- //抽象构件:物品
- interface Articles {
- public float calculation(); //计算
- public void show();
- }
- //树叶构件:商品
- class Goods implements Articles {
- private String name; //名字
- private int quantity; //数量
- private float unitPrice; //单价
- public Goods(String name, int quantity, float unitPrice) {
- this.name = name;
- this.quantity = quantity;
- this.unitPrice = unitPrice;
- }
- public float calculation() {
- return quantity * unitPrice;
- }
- public void show() {
- System.out.println(name + "(数量:" + quantity + ",单价:" + unitPrice + "元)");
- }
- }
- //树枝构件:袋子
- class Bags implements Articles {
- private String name; //名字
- private ArrayList<Articles> bags = new ArrayList<Articles>();
- public Bags(String name) {
- this.name = name;
- }
- public void add(Articles c) {
- bags.add(c);
- }
- public void remove(Articles c) {
- bags.remove(c);
- }
- public Articles getChild(int i) {
- return bags.get(i);
- }
- public float calculation() {
- float s = 0;
- for (Object obj : bags) {
- s += ((Articles) obj).calculation();
- }
- return s;
- }
- public void show() {
- for (Object obj : bags) {
- ((Articles) obj).show();
- }
- }
- }
程序运行结果如下:
您选购的商品有: 李宁牌运动鞋(数量:1,单价:198.0元) 韶关香菇(数量:2,单价:68.0元) 韶关红茶(数量:3,单价:180.0元) 景德镇瓷器(数量:1,单价:380.0元) 婺源特产(数量:2,单价:7.9元) 婺源地图(数量:1,单价:9.9元) 要支付的总价是:1279.7元
前面分析了组合模式的结构与特点,下面分析它适用的以下应用场景。
如果对前面介绍的组合模式中的树叶节点和树枝节点进行抽象,也就是说树叶节点和树枝节点还有子节点,这时组合模式就扩展成复杂的组合模式了,如 Java AWT/Swing 中的简单组件 JTextComponent 有子类 JTextField、JTextArea,容器组件 Container 也有子类 Window、Panel。复杂的组合模式的结构图如图 5 所示。
图5 复杂的组合模式的结构图
结构型模式(Structural Pattern)描述如何将类或者对象结合在一起形成更大的结构,就像搭积木, 可以通过简单积木的组合形成复杂的、功能更为强大的结构。结构型模式可以分为类结构型模式和对象结构型模式,也可分为代理模式(Proxy)、适配器模式(Adapter)、桥接模式(Bridge)、装饰模式 (Decorator )、外观模式(Facade)、享元模式(Flyweight)和组合模式(Composite)等 7 类。
本实验的主要目的如下。
1. 结构型模式的工作原理
结构型模式重点考虑类或对象的布局方式,其目的是将现有类或对象组成更大的结构。按照其显示方式的不同,结构型模式可分为类结构型模式和对象结构型模式。前者采用继承机制来组织接口和类,后者采用组合或聚合来组合对象。
由于组合关系和或聚合关系比继承关系耦合度低,满足“合成复用原则”,所以对象结构型模式比类结构型模式具有更大的灵活性。如果按目的来分,结构型模式共 7 种,每种模式的实验大概要花 2 个学时,大家可以根据实验计划来选做若干个实验。下面以代理(Proxy)模式为例,介绍其实验过程。
2. 代理模式的工作原理
代理模式是在访问对象和目标对象之间增加一个代理对象,该对象起到中介作用和保护目标对象的作用。另外,它还可以扩展目标对象的功能,并且将客户端与目标对象分离,这在一定程度上降低了系统的耦合度。
代理模式的结构比较简单,主要是通过定义一个继承抽象主题的代理来包含真实主题,从而实现对真实主题的访问,其结构图如图 1 所示。
图1 代理模式的结构图
代理模式的主要角色如下。
(1) 用代理模式设计一个房产中介的模拟程序。
要求:房产中介有介绍和代售韶关碧桂园房子的权利,以及登记购房者信息权利。这里的房产中介是代理者,韶关碧桂园是真实主题。
(2) 按照以上要求设计类图和编写 Java 源程序。
所设计的实验程序要满足以下两点。
(1) 用 UML 设计“房产中介”模拟程序的结构图。
“房产中介”模拟程序的结构图如图 2 所示。
图2 房产中介模拟程序的结构图
(2) 根据结构图写出“房产中介”模拟程序的源代码。
房产中介模拟程序的源代码如下。
- package proxy;
- import java.awt.*;
- import javax.swing.*;
- public class SgHouseProxy {
- public static void main(String[] args) {
- HouseProxy proxy = new HouseProxy();
- proxy.display();
- }
- }
- //抽象主题:房主
- interface HouseOwner {
- void display();
- }
- //真实主题:韶关碧桂园
- class SgBiguiyuan extends JFrame implements HouseOwner {
- private static final long serialVersionUID = 1L;
- public SgBiguiyuan() {
- super("房产中介代售韶关碧桂园房子");
- }
- public void display() {
- this.setLayout(new GridLayout(1, 1));
- JLabel l1 = new JLabel(new ImageIcon("src/proxy/SgBiguiyuan.jpg"));
- this.add(l1);
- this.pack();
- this.setVisible(true);
- this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
- }
- }
- //代理:房产中介
- class HouseProxy implements HouseOwner {
- private SgBiguiyuan realSubject = new SgBiguiyuan();
- public void display() {
- preRequest();
- realSubject.display();
- postRequest();
- }
- public void preRequest() {
- System.out.println("房产中介介绍韶关碧桂园房子。");
- }
- public void postRequest() {
- System.out.println("房产中介登记购房者信息。");
- }
- }
(3) 上机测试程序,查看运行结果。
“房产中介”模拟程序的运行结果如图 3 所示。
图3 房产中介模拟程序的运行结果
(4) 按同样的步骤设计其他“结构型模式”的程序实例。
行为型模式用于描述程序在运行时复杂的流程控制,即描述多个类或对象之间怎样相互协作共同完成单个对象都无法单独完成的任务,它涉及算法与对象间职责的分配。
行为型模式分为类行为模式和对象行为模式,前者采用继承机制来在类间分派行为,后者采用组合或聚合在对象间分配行为。由于组合关系或聚合关系比继承关系耦合度低,满足“合成复用原则”,所以对象行为模式比类行为模式具有更大的灵活性。
行为型模式是 GoF 设计模式中最为庞大的一类,它包含以下 11 种模式。
以上 11 种行为型模式,除了模板方法模式和解释器模式是类行为型模式,其他的全部属于对象行为型模式,下面我们将详细介绍它们的特点、结构与应用。
在面向对象程序设计过程中,程序员常常会遇到这种情况:设计一个系统时知道了算法所需的关键步骤,而且确定了这些步骤的执行顺序,但某些步骤的具体实现还未知,或者说某些步骤的实现与具体的环境相关。
例如,去银行办理业务一般要经过以下4个流程:取号、排队、办理具体业务、对银行工作人员进行评分等,其中取号、排队和对银行工作人员进行评分的业务对每个客户是一样的,可以在父类中实现,但是办理具体业务却因人而异,它可能是存款、取款或者转账等,可以延迟到子类中实现。
这样的例子在生活中还有很多,例如,一个人每天会起床、吃饭、做事、睡觉等,其中“做事”的内容每天可能不同。我们把这些规定了流程或格式的实例定义成模板,允许使用者根据自己的需求去更新它,例如,简历模板、论文模板、Word 中模板文件等。
以下介绍的模板方法模式将解决以上类似的问题。
模板方法(Template Method)模式的定义如下:定义一个操作中的算法骨架,而将算法的一些步骤延迟到子类中,使得子类可以不改变该算法结构的情况下重定义该算法的某些特定步骤。它是一种类行为型模式。
该模式的主要优点如下。
该模式的主要缺点如下。
模板方法模式需要注意抽象类与具体子类之间的协作。它用到了虚函数的多态性技术以及“不用调用我,让我来调用你”的反向控制技术。现在来介绍它们的基本结构。
模板方法模式包含以下主要角色。
1)抽象类/抽象模板(Abstract Class)
抽象模板类,负责给出一个算法的轮廓和骨架。它由一个模板方法和若干个基本方法构成。这些方法的定义如下。
① 模板方法:定义了算法的骨架,按某种顺序调用其包含的基本方法。
② 基本方法:是整个算法中的一个步骤,包含以下几种类型。
2)具体子类/具体实现(Concrete Class)
具体实现类,实现抽象类中所定义的抽象方法和钩子方法,它们是一个顶级逻辑的一个组成步骤。
模板方法模式的结构图如图 1 所示。
图1 模板方法模式的结构图
模板方法模式的代码如下:
- public class TemplateMethodPattern {
- public static void main(String[] args) {
- AbstractClass tm = new ConcreteClass();
- tm.TemplateMethod();
- }
- }
- //抽象类
- abstract class AbstractClass {
- //模板方法
- public void TemplateMethod() {
- SpecificMethod();
- abstractMethod1();
- abstractMethod2();
- }
- //具体方法
- public void SpecificMethod() {
- System.out.println("抽象类中的具体方法被调用...");
- }
- //抽象方法1
- public abstract void abstractMethod1();
- //抽象方法2
- public abstract void abstractMethod2();
- }
- //具体子类
- class ConcreteClass extends AbstractClass {
- public void abstractMethod1() {
- System.out.println("抽象方法1的实现被调用...");
- }
- public void abstractMethod2() {
- System.out.println("抽象方法2的实现被调用...");
- }
- }
程序的运行结果如下:
抽象类中的具体方法被调用... 抽象方法1的实现被调用... 抽象方法2的实现被调用...
【例1】用模板方法模式实现出国留学手续设计程序。
分析:出国留学手续一般经过以下流程:索取学校资料,提出入学申请,办理因私出国护照、出境卡和公证,申请签证,体检、订机票、准备行装,抵达目标学校等,其中有些业务对各个学校是一样的,但有些业务因学校不同而不同,所以比较适合用模板方法模式来实现。
在本实例中,我们先定义一个出国留学的抽象类 StudyAbroad,里面包含了一个模板方法 TemplateMethod(),该方法中包含了办理出国留学手续流程中的各个基本方法,其中有些方法的处理由于各国都一样,所以在抽象类中就可以实现,但有些方法的处理各国是不同的,必须在其具体子类(如美国留学类 StudyInAmerica)中实现。如果再增加一个国家,只要增加一个子类就可以了,图 2 所示是其结构图。
图2 出国留学手续设计程序的结构图
程序代码如下:
- public class StudyAbroadProcess {
- public static void main(String[] args) {
- StudyAbroad tm = new StudyInAmerica();
- tm.TemplateMethod();
- }
- }
- //抽象类: 出国留学
- abstract class StudyAbroad {
- public void TemplateMethod() //模板方法
- {
- LookingForSchool(); //索取学校资料
- ApplyForEnrol(); //入学申请
- ApplyForPassport(); //办理因私出国护照、出境卡和公证
- ApplyForVisa(); //申请签证
- ReadyGoAbroad(); //体检、订机票、准备行装
- Arriving(); //抵达
- }
- public void ApplyForPassport() {
- System.out.println("三.办理因私出国护照、出境卡和公证:");
- System.out.println(" 1)持录取通知书、本人户口簿或身份证向户口所在地公安机关申请办理因私出国护照和出境卡。");
- System.out.println(" 2)办理出生公证书,学历、学位和成绩公证,经历证书,亲属关系公证,经济担保公证。");
- }
- public void ApplyForVisa() {
- System.out.println("四.申请签证:");
- System.out.println(" 1)准备申请国外境签证所需的各种资料,包括个人学历、成绩单、工作经历的证明;个人及家庭收入、资金和财产证明;家庭成员的关系证明等;");
- System.out.println(" 2)向拟留学国家驻华使(领)馆申请入境签证。申请时需按要求填写有关表格,递交必需的证明材料,缴纳签证。有的国家(比如美国、英国、加拿大等)在申请签证时会要求申请人前往使(领)馆进行面试。");
- }
- public void ReadyGoAbroad() {
- System.out.println("五.体检、订机票、准备行装:");
- System.out.println(" 1)进行身体检查、免疫检查和接种传染病疫苗;");
- System.out.println(" 2)确定机票时间、航班和转机地点。");
- }
- public abstract void LookingForSchool();//索取学校资料
- public abstract void ApplyForEnrol(); //入学申请
- public abstract void Arriving(); //抵达
- }
- //具体子类: 美国留学
- class StudyInAmerica extends StudyAbroad {
- @Override
- public void LookingForSchool() {
- System.out.println("一.索取学校以下资料:");
- System.out.println(" 1)对留学意向国家的政治、经济、文化背景和教育体制、学术水平进行较为全面的了解;");
- System.out.println(" 2)全面了解和掌握国外学校的情况,包括历史、学费、学制、专业、师资配备、教学设施、学术地位、学生人数等;");
- System.out.println(" 3)了解该学校的住宿、交通、医疗保险情况如何;");
- System.out.println(" 4)该学校在中国是否有授权代理招生的留学中介公司?");
- System.out.println(" 5)掌握留学签证情况;");
- System.out.println(" 6)该国政府是否允许留学生合法打工?");
- System.out.println(" 8)毕业之后可否移民?");
- System.out.println(" 9)文凭是否受到我国认可?");
- }
- @Override
- public void ApplyForEnrol() {
- System.out.println("二.入学申请:");
- System.out.println(" 1)填写报名表;");
- System.out.println(" 2)将报名表、个人学历证明、最近的学习成绩单、推荐信、个人简历、托福或雅思语言考试成绩单等资料寄往所申请的学校;");
- System.out.println(" 3)为了给签证办理留有充裕的时间,建议越早申请越好,一般提前1年就比较从容。");
- }
- @Override
- public void Arriving() {
- System.out.println("六.抵达目标学校:");
- System.out.println(" 1)安排住宿;");
- System.out.println(" 2)了解校园及周边环境。");
- }
- }
程序的运行结果如下:
一.索取学校以下资料: 1)对留学意向国家的政治、经济、文化背景和教育体制、学术水平进行较为全面的了解; 2)全面了解和掌握国外学校的情况,包括历史、学费、学制、专业、师资配备、教学设施、学术地位、学生人数等; 3)了解该学校的住宿、交通、医疗保险情况如何; 4)该学校在中国是否有授权代理招生的留学中介公司? 5)掌握留学签证情况; 6)该国政府是否允许留学生合法打工? 8)毕业之后可否移民? 9)文凭是否受到我国认可? 二.入学申请: 1)填写报名表; 2)将报名表、个人学历证明、最近的学习成绩单、推荐信、个人简历、托福或雅思语言考试成绩单等资料寄往所申请的学校; 3)为了给签证办理留有充裕的时间,建议越早申请越好,一般提前1年就比较从容。 三.办理因私出国护照、出境卡和公证: 1)持录取通知书、本人户口簿或身份证向户口所在地公安机关申请办理因私出国护照和出境卡。 2)办理出生公证书,学历、学位和成绩公证,经历证书,亲属关系公证,经济担保公证。 四.申请签证: 1)准备申请国外境签证所需的各种资料,包括个人学历、成绩单、工作经历的证明;个人及家庭收入、资金和财产证明;家庭成员的关系证明等; 2)向拟留学国家驻华使(领)馆申请入境签证。申请时需按要求填写有关表格,递交必需的证明材料,缴纳签证。有的国家(比如美国、英国、加拿大等)在申请签证时会要求申请人前往使(领)馆进行面试。 五.体检、订机票、准备行装: 1)进行身体检查、免疫检查和接种传染病疫苗; 2)确定机票时间、航班和转机地点。 六.抵达目标学校: 1)安排住宿; 2)了解校园及周边环境。
模板方法模式通常适用于以下场景。
在模板方法模式中,基本方法包含:抽象方法、具体方法和钩子方法,正确使用“钩子方法”可以使得子类控制父类的行为。如下面例子中,可以通过在具体子类中重写钩子方法 HookMethod1() 和 HookMethod2() 来改变抽象父类中的运行结果,其结构图如图 3 所示。
图3 含钩子方法的模板方法模式的结构图
程序代码如下:
- public class HookTemplateMethod {
- public static void main(String[] args) {
- HookAbstractClass tm = new HookConcreteClass();
- tm.TemplateMethod();
- }
- }
- //含钩子方法的抽象类
- abstract class HookAbstractClass {
- //模板方法
- public void TemplateMethod() {
- abstractMethod1();
- HookMethod1();
- if (HookMethod2()) {
- SpecificMethod();
- }
- abstractMethod2();
- }
- //具体方法
- public void SpecificMethod() {
- System.out.println("抽象类中的具体方法被调用...");
- }
- //钩子方法1
- public void HookMethod1() {
- }
- //钩子方法2
- public boolean HookMethod2() {
- return true;
- }
- //抽象方法1
- public abstract void abstractMethod1();
- //抽象方法2
- public abstract void abstractMethod2();
- }
- //含钩子方法的具体子类
- class HookConcreteClass extends HookAbstractClass {
- public void abstractMethod1() {
- System.out.println("抽象方法1的实现被调用...");
- }
- public void abstractMethod2() {
- System.out.println("抽象方法2的实现被调用...");
- }
- public void HookMethod1() {
- System.out.println("钩子方法1被重写...");
- }
- public boolean HookMethod2() {
- return false;
- }
- }
程序的运行结果如下:
抽象方法1的实现被调用... 钩子方法1被重写... 抽象方法2的实现被调用...
如果钩子方法 HookMethod1() 和钩子方法 HookMethod2() 的代码改变,则程序的运行结果也会改变。
在现实生活中常常遇到实现某种目标存在多种策略可供选择的情况,例如,出行旅游可以乘坐飞机、乘坐火车、骑自行车或自己开私家车等,超市促销可以釆用打折、送商品、送积分等方法。
在软件开发中也常常遇到类似的情况,当实现某一个功能存在多种算法或者策略,我们可以根据环境或者条件的不同选择不同的算法或者策略来完成该功能,如数据排序策略有冒泡排序、选择排序、插入排序、二叉树排序等。
如果使用多重条件转移语句实现(即硬编码),不但使条件语句变得很复杂,而且增加、删除或更换算法要修改原代码,不易维护,违背开闭原则。如果采用策略模式就能很好解决该问题。
策略(Strategy)模式的定义:该模式定义了一系列算法,并将每个算法封装起来,使它们可以相互替换,且算法的变化不会影响使用算法的客户。策略模式属于对象行为模式,它通过对算法进行封装,把使用算法的责任和算法的实现分割开来,并委派给不同的对象对这些算法进行管理。
策略模式的主要优点如下。
其主要缺点如下。
策略模式是准备一组算法,并将这组算法封装到一系列的策略类里面,作为一个抽象策略类的子类。策略模式的重心不是如何实现算法,而是如何组织这些算法,从而让程序结构更加灵活,具有更好的维护性和扩展性,现在我们来分析其基本结构和实现方法。
1. 模式的结构
策略模式的主要角色如下。
其结构图如图 1 所示。
图1 策略模式的结构图
2. 模式的实现
策略模式的实现代码如下:
- public class StrategyPattern {
- public static void main(String[] args) {
- Context c = new Context();
- Strategy s = new ConcreteStrategyA();
- c.setStrategy(s);
- c.strategyMethod();
- System.out.println("-----------------");
- s = new ConcreteStrategyB();
- c.setStrategy(s);
- c.strategyMethod();
- }
- }
- //抽象策略类
- interface Strategy {
- public void strategyMethod(); //策略方法
- }
- //具体策略类A
- class ConcreteStrategyA implements Strategy {
- public void strategyMethod() {
- System.out.println("具体策略A的策略方法被访问!");
- }
- }
- //具体策略类B
- class ConcreteStrategyB implements Strategy {
- public void strategyMethod() {
- System.out.println("具体策略B的策略方法被访问!");
- }
- }
- //环境类
- class Context {
- private Strategy strategy;
- public Strategy getStrategy() {
- return strategy;
- }
- public void setStrategy(Strategy strategy) {
- this.strategy = strategy;
- }
- public void strategyMethod() {
- strategy.strategyMethod();
- }
- }
程序运行结果如下:
具体策略A的策略方法被访问! ----------------- 具体策略B的策略方法被访问!
【例1】策略模式在“大闸蟹”做菜中的应用。
分析:关于大闸蟹的做法有很多种,我们以清蒸大闸蟹和红烧大闸蟹两种方法为例,介绍策略模式的应用。
首先,定义一个大闸蟹加工的抽象策略类(CrabCooking),里面包含了一个做菜的抽象方法 CookingMethod();然后,定义清蒸大闸蟹(SteamedCrabs)和红烧大闸蟹(BraisedCrabs)的具体策略类,它们实现了抽象策略类中的抽象方法;由于本程序要显示做好的结果图(点此下载要显示的结果图),所以将具体策略类定义成 JLabel 的子类;最后,定义一个厨房(Kitchen)环境类,它具有设置和选择做菜策略的方法;客户类通过厨房类获取做菜策略,并把做菜结果图在窗体中显示出来,图 2 所示是其结构图。
图2 大闸蟹做菜策略的结构图
程序代码如下:
- import java.awt.*;
- import java.awt.event.*;
- import javax.swing.*;
- public class CrabCookingStrategy implements ItemListener {
- private JFrame f;
- private JRadioButton qz, hs;
- private JPanel CenterJP, SouthJP;
- private Kitchen cf; //厨房
- private CrabCooking qzx, hsx; //大闸蟹加工者
- CrabCookingStrategy() {
- f = new JFrame("策略模式在大闸蟹做菜中的应用");
- f.setBounds(100, 100, 500, 400);
- f.setVisible(true);
- f.setResizable(false);
- f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
- SouthJP = new JPanel();
- CenterJP = new JPanel();
- f.add("South", SouthJP);
- f.add("Center", CenterJP);
- qz = new JRadioButton("清蒸大闸蟹");
- hs = new JRadioButton("红烧大闸蟹");
- qz.addItemListener(this);
- hs.addItemListener(this);
- ButtonGroup group = new ButtonGroup();
- group.add(qz);
- group.add(hs);
- SouthJP.add(qz);
- SouthJP.add(hs);
- //---------------------------------
- cf = new Kitchen(); //厨房
- qzx = new SteamedCrabs(); //清蒸大闸蟹类
- hsx = new BraisedCrabs(); //红烧大闸蟹类
- }
- public void itemStateChanged(ItemEvent e) {
- JRadioButton jc = (JRadioButton) e.getSource();
- if (jc == qz) {
- cf.setStrategy(qzx);
- cf.CookingMethod(); //清蒸
- } else if (jc == hs) {
- cf.setStrategy(hsx);
- cf.CookingMethod(); //红烧
- }
- CenterJP.removeAll();
- CenterJP.repaint();
- CenterJP.add((Component) cf.getStrategy());
- f.setVisible(true);
- }
- public static void main(String[] args) {
- new CrabCookingStrategy();
- }
- }
- //抽象策略类:大闸蟹加工类
- interface CrabCooking {
- public void CookingMethod(); //做菜方法
- }
- //具体策略类:清蒸大闸蟹
- class SteamedCrabs extends JLabel implements CrabCooking {
- private static final long serialVersionUID = 1L;
- public void CookingMethod() {
- this.setIcon(new ImageIcon("src/strategy/SteamedCrabs.jpg"));
- this.setHorizontalAlignment(CENTER);
- }
- }
- //具体策略类:红烧大闸蟹
- class BraisedCrabs extends JLabel implements CrabCooking {
- private static final long serialVersionUID = 1L;
- public void CookingMethod() {
- this.setIcon(new ImageIcon("src/strategy/BraisedCrabs.jpg"));
- this.setHorizontalAlignment(CENTER);
- }
- }
- //环境类:厨房
- class Kitchen {
- private CrabCooking strategy; //抽象策略
- public void setStrategy(CrabCooking strategy) {
- this.strategy = strategy;
- }
- public CrabCooking getStrategy() {
- return strategy;
- }
- public void CookingMethod() {
- strategy.CookingMethod(); //做菜
- }
- }
程序运行结果如图 3 所示。
图3 大闸蟹做菜结果
【例2】用策略模式实现从韶关去婺源旅游的出行方式。
分析:从韶关去婺源旅游有以下几种出行方式:坐火车、坐汽车和自驾车,所以该实例用策略模式比较适合,图 4 所示是其结构图。
图4 婺源旅游结构图
策略模式在很多地方用到,如 Java SE 中的容器布局管理就是一个典型的实例,Java SE 中的每个容器都存在多种布局供用户选择。在程序设计中,通常在以下几种情况中使用策略模式较多。
在一个使用策略模式的系统中,当存在的策略很多时,客户端管理所有策略算法将变得很复杂,如果在环境类中使用策略工厂模式来管理这些策略类将大大减少客户端的工作复杂度,其结构图如图 5 所示。
图5 策略工厂模式的结构图
在软件开发系统中,“方法的请求者”与“方法的实现者”之间经常存在紧密的耦合关系,这不利于软件功能的扩展与维护。例如,想对方法进行“撤销、重做、记录”等处理都很不方便,因此“如何将方法的请求者与实现者解耦?”变得很重要,命令模式就能很好地解决这个问题。
在现实生活中,命令模式的例子也很多。比如看电视时,我们只需要轻轻一按遥控器就能完成频道的切换,这就是命令模式,将换台请求和换台处理完全解耦了。电视机遥控器(命令发送者)通过按钮(具体命令)来遥控电视机(命令接收者)。
再比如,我们去餐厅吃饭,菜单不是等到客人来了之后才定制的,而是已经预先配置好的。这样,客人来了就只需要点菜,而不是任由客人临时定制。餐厅提供的菜单就相当于把请求和处理进行了解耦,这就是命令模式的体现。
命令(Command)模式的定义如下:将一个请求封装为一个对象,使发出请求的责任和执行请求的责任分割开。这样两者之间通过命令对象进行沟通,这样方便将命令对象进行储存、传递、调用、增加与管理。
命令模式的主要优点如下。
其缺点是:
可以将系统中的相关操作抽象成命令,使调用者与实现者相关分离,其结构如下。
1. 模式的结构
命令模式包含以下主要角色。
其结构图如图 1 所示。
图1 命令模式的结构图
2. 模式的实现
命令模式的代码如下:
- package command;
- public class CommandPattern {
- public static void main(String[] args) {
- Command cmd = new ConcreteCommand();
- Invoker ir = new Invoker(cmd);
- System.out.println("客户访问调用者的call()方法...");
- ir.call();
- }
- }
- //调用者
- class Invoker {
- private Command command;
- public Invoker(Command command) {
- this.command = command;
- }
- public void setCommand(Command command) {
- this.command = command;
- }
- public void call() {
- System.out.println("调用者执行命令command...");
- command.execute();
- }
- }
- //抽象命令
- interface Command {
- public abstract void execute();
- }
- //具体命令
- class ConcreteCommand implements Command {
- private Receiver receiver;
- ConcreteCommand() {
- receiver = new Receiver();
- }
- public void execute() {
- receiver.action();
- }
- }
- //接收者
- class Receiver {
- public void action() {
- System.out.println("接收者的action()方法被调用...");
- }
- }
程序的运行结果如下:
客户访问调用者的call()方法... 调用者执行命令command... 接收者的action()方法被调用...
【例1】用命令模式实现客户去餐馆吃早餐的实例。
分析:客户去餐馆可选择的早餐有肠粉、河粉和馄饨等,客户可向服务员选择以上早餐中的若干种,服务员将客户的请求交给相关的厨师去做。这里的点早餐相当于“命令”,服务员相当于“调用者”,厨师相当于“接收者”,所以用命令模式实现比较合适。
由于本实例要显示厨师做菜的效果图(点此下载要显示的效果图),所以把每个厨师类定义为 JFrame 的子类;最后,定义服务员类(Waiter),它接收客户的做菜请求,并发出做菜的命令。客户类是通过服务员类来点菜的,图 2 所示是其结构图。
图2 客户在餐馆吃早餐的结构图
程序代码如下:
- package command;
- import javax.swing.*;
- public class CookingCommand {
- public static void main(String[] args) {
- Breakfast food1 = new ChangFen();
- Breakfast food2 = new HunTun();
- Breakfast food3 = new HeFen();
- Waiter fwy = new Waiter();
- fwy.setChangFen(food1);//设置肠粉菜单
- fwy.setHunTun(food2); //设置河粉菜单
- fwy.setHeFen(food3); //设置馄饨菜单
- fwy.chooseChangFen(); //选择肠粉
- fwy.chooseHeFen(); //选择河粉
- fwy.chooseHunTun(); //选择馄饨
- }
- }
- //调用者:服务员
- class Waiter {
- private Breakfast changFen, hunTun, heFen;
- public void setChangFen(Breakfast f) {
- changFen = f;
- }
- public void setHunTun(Breakfast f) {
- hunTun = f;
- }
- public void setHeFen(Breakfast f) {
- heFen = f;
- }
- public void chooseChangFen() {
- changFen.cooking();
- }
- public void chooseHunTun() {
- hunTun.cooking();
- }
- public void chooseHeFen() {
- heFen.cooking();
- }
- }
- //抽象命令:早餐
- interface Breakfast {
- public abstract void cooking();
- }
- //具体命令:肠粉
- class ChangFen implements Breakfast {
- private ChangFenChef receiver;
- ChangFen() {
- receiver = new ChangFenChef();
- }
- public void cooking() {
- receiver.cooking();
- }
- }
- //具体命令:馄饨
- class HunTun implements Breakfast {
- private HunTunChef receiver;
- HunTun() {
- receiver = new HunTunChef();
- }
- public void cooking() {
- receiver.cooking();
- }
- }
- //具体命令:河粉
- class HeFen implements Breakfast {
- private HeFenChef receiver;
- HeFen() {
- receiver = new HeFenChef();
- }
- public void cooking() {
- receiver.cooking();
- }
- }
- //接收者:肠粉厨师
- class ChangFenChef extends JFrame {
- private static final long serialVersionUID = 1L;
- JLabel l = new JLabel();
- ChangFenChef() {
- super("煮肠粉");
- l.setIcon(new ImageIcon("src/command/ChangFen.jpg"));
- this.add(l);
- this.setLocation(30, 30);
- this.pack();
- this.setResizable(false);
- this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
- }
- public void cooking() {
- this.setVisible(true);
- }
- }
- //接收者:馄饨厨师
- class HunTunChef extends JFrame {
- private static final long serialVersionUID = 1L;
- JLabel l = new JLabel();
- HunTunChef() {
- super("煮馄饨");
- l.setIcon(new ImageIcon("src/command/HunTun.jpg"));
- this.add(l);
- this.setLocation(350, 50);
- this.pack();
- this.setResizable(false);
- this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
- }
- public void cooking() {
- this.setVisible(true);
- }
- }
- //接收者:河粉厨师
- class HeFenChef extends JFrame {
- private static final long serialVersionUID = 1L;
- JLabel l = new JLabel();
- HeFenChef() {
- super("煮河粉");
- l.setIcon(new ImageIcon("src/command/HeFen.jpg"));
- this.add(l);
- this.setLocation(200, 280);
- this.pack();
- this.setResizable(false);
- this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
- }
- public void cooking() {
- this.setVisible(true);
- }
- }
程序的运行结果如图 2 所示。
图2 客户在餐馆吃早餐的运行结果
当系统的某项操作具备命令语义,且命令实现不稳定(变化)时,可以通过命令模式解耦请求与实现。使用抽象命令接口使请求方的代码架构稳定,封装接收方具体命令的实现细节。接收方与抽象命令呈现弱耦合(内部方法无需一致),具备良好的扩展性。
命令模式通常适用于以下场景。
在软件开发中,有时将命令模式与前面学的组合模式联合使用,这就构成了宏命令模式,也叫组合命令模式。宏命令包含了一组命令,它充当了具体命令与调用者的双重角色,执行它时将递归调用它所包含的所有命令,其具体结构图如图 3 所示。
图3 组合命令模式的结构图
程序代码如下:
- package command;
- import java.util.ArrayList;
- public class CompositeCommandPattern {
- public static void main(String[] args) {
- AbstractCommand cmd1 = new ConcreteCommand1();
- AbstractCommand cmd2 = new ConcreteCommand2();
- CompositeInvoker ir = new CompositeInvoker();
- ir.add(cmd1);
- ir.add(cmd2);
- System.out.println("客户访问调用者的execute()方法...");
- ir.execute();
- }
- }
- //抽象命令
- interface AbstractCommand {
- public abstract void execute();
- }
- //树叶构件: 具体命令1
- class ConcreteCommand1 implements AbstractCommand {
- private CompositeReceiver receiver;
- ConcreteCommand1() {
- receiver = new CompositeReceiver();
- }
- public void execute() {
- receiver.action1();
- }
- }
- //树叶构件: 具体命令2
- class ConcreteCommand2 implements AbstractCommand {
- private CompositeReceiver receiver;
- ConcreteCommand2() {
- receiver = new CompositeReceiver();
- }
- public void execute() {
- receiver.action2();
- }
- }
- //树枝构件: 调用者
- class CompositeInvoker implements AbstractCommand {
- private ArrayList<AbstractCommand> children = new ArrayList<AbstractCommand>();
- public void add(AbstractCommand c) {
- children.add(c);
- }
- public void remove(AbstractCommand c) {
- children.remove(c);
- }
- public AbstractCommand getChild(int i) {
- return children.get(i);
- }
- public void execute() {
- for (Object obj : children) {
- ((AbstractCommand) obj).execute();
- }
- }
- }
- //接收者
- class CompositeReceiver {
- public void action1() {
- System.out.println("接收者的action1()方法被调用...");
- }
- public void action2() {
- System.out.println("接收者的action2()方法被调用...");
- }
- }
程序的运行结果如下:
客户访问调用者的execute()方法... 接收者的action1()方法被调用... 接收者的action2()方法被调用...
当然,命令模式还可以同备忘录(Memento)模式组合使用,这样就变成了可撤销的命令模式,这将在后面介绍。
在现实生活中,一个事件需要经过多个对象处理是很常见的场景。例如,采购审批流程、请假流程等。公司员工请假,可批假的领导有部门负责人、副总经理、总经理等,但每个领导能批准的天数不同,员工必须根据需要请假的天数去找不同的领导签名,也就是说员工必须记住每个领导的姓名、电话和地址等信息,这无疑增加了难度。
在计算机软硬件中也有相关例子,如总线网中数据报传送,每台计算机根据目标地址是否同自己的地址相同来决定是否接收;还有异常处理中,处理程序根据异常的类型决定自己是否处理该异常;还有 Struts2 的拦截器、JSP 和 Servlet 的 Filter 等,所有这些,都可以考虑使用责任链模式来实现。
责任链(Chain of Responsibility)模式的定义:为了避免请求发送者与多个请求处理者耦合在一起,于是将所有请求的处理者通过前一对象记住其下一个对象的引用而连成一条链;当有请求发生时,可将请求沿着这条链传递,直到有对象处理它为止。
注意:责任链模式也叫职责链模式。
在责任链模式中,客户只需要将请求发送到责任链上即可,无须关心请求的处理细节和请求的传递过程,请求会自动进行传递。所以责任链将请求的发送者和请求的处理者解耦了。
责任链模式是一种对象行为型模式,其主要优点如下。
其主要缺点如下。
通常情况下,可以通过数据链表来实现职责链模式的数据结构。
1. 模式的结构
职责链模式主要包含以下角色。
责任链模式的本质是解耦请求与处理,让请求在处理链中能进行传递与被处理;理解责任链模式应当理解其模式,而不是其具体实现。责任链模式的独到之处是将其节点处理者组合成了链式结构,并允许节点自身决定是否进行请求处理或转发,相当于让请求流动起来。
其结构图如图 1 所示。客户端可按图 2 所示设置责任链。
图1 责任链模式的结构图
图2 责任链
2. 模式的实现
职责链模式的实现代码如下:
- package chainOfResponsibility;
- public class ChainOfResponsibilityPattern {
- public static void main(String[] args) {
- //组装责任链
- Handler handler1 = new ConcreteHandler1();
- Handler handler2 = new ConcreteHandler2();
- handler1.setNext(handler2);
- //提交请求
- handler1.handleRequest("two");
- }
- }
- //抽象处理者角色
- abstract class Handler {
- private Handler next;
- public void setNext(Handler next) {
- this.next = next;
- }
- public Handler getNext() {
- return next;
- }
- //处理请求的方法
- public abstract void handleRequest(String request);
- }
- //具体处理者角色1
- class ConcreteHandler1 extends Handler {
- public void handleRequest(String request) {
- if (request.equals("one")) {
- System.out.println("具体处理者1负责处理该请求!");
- } else {
- if (getNext() != null) {
- getNext().handleRequest(request);
- } else {
- System.out.println("没有人处理该请求!");
- }
- }
- }
- }
- //具体处理者角色2
- class ConcreteHandler2 extends Handler {
- public void handleRequest(String request) {
- if (request.equals("two")) {
- System.out.println("具体处理者2负责处理该请求!");
- } else {
- if (getNext() != null) {
- getNext().handleRequest(request);
- } else {
- System.out.println("没有人处理该请求!");
- }
- }
- }
- }
程序运行结果如下:
具体处理者2负责处理该请求!
在上面代码中,我们把消息硬编码为 String 类型,而在真实业务中,消息是具备多样性的,可以是 int、String 或者自定义类型。因此,在上面代码的基础上,可以对消息类型进行抽象 Request,增强了消息的兼容性。
【例1】用责任链模式设计一个请假条审批模块。
分析:假如规定学生请假小于或等于 2 天,班主任可以批准;小于或等于 7 天,系主任可以批准;小于或等于 10 天,院长可以批准;其他情况不予批准;这个实例适合使用职责链模式实现。
首先,定义一个领导类(Leader),它是抽象处理者,包含了一个指向下一位领导的指针 next 和一个处理假条的抽象处理方法 handleRequest(int LeaveDays);然后,定义班主任类(ClassAdviser)、系主任类(DepartmentHead)和院长类(Dean),它们是抽象处理者的子类,是具体处理者,必须根据自己的权力去实现父类的 handleRequest(int LeaveDays) 方法,如果无权处理就将假条交给下一位具体处理者,直到最后;客户类负责创建处理链,并将假条交给链头的具体处理者(班主任)。图 3 所示是其结构图。
图3 请假条审批模块的结构图
程序代码如下:
- package chainOfResponsibility;
- public class LeaveApprovalTest {
- public static void main(String[] args) {
- //组装责任链
- Leader teacher1 = new ClassAdviser();
- Leader teacher2 = new DepartmentHead();
- Leader teacher3 = new Dean();
- //Leader teacher4=new DeanOfStudies();
- teacher1.setNext(teacher2);
- teacher2.setNext(teacher3);
- //teacher3.setNext(teacher4);
- //提交请求
- teacher1.handleRequest(8);
- }
- }
- //抽象处理者:领导类
- abstract class Leader {
- private Leader next;
- public void setNext(Leader next) {
- this.next = next;
- }
- public Leader getNext() {
- return next;
- }
- //处理请求的方法
- public abstract void handleRequest(int LeaveDays);
- }
- //具体处理者1:班主任类
- class ClassAdviser extends Leader {
- public void handleRequest(int LeaveDays) {
- if (LeaveDays <= 2) {
- System.out.println("班主任批准您请假" + LeaveDays + "天。");
- } else {
- if (getNext() != null) {
- getNext().handleRequest(LeaveDays);
- } else {
- System.out.println("请假天数太多,没有人批准该假条!");
- }
- }
- }
- }
- //具体处理者2:系主任类
- class DepartmentHead extends Leader {
- public void handleRequest(int LeaveDays) {
- if (LeaveDays <= 7) {
- System.out.println("系主任批准您请假" + LeaveDays + "天。");
- } else {
- if (getNext() != null) {
- getNext().handleRequest(LeaveDays);
- } else {
- System.out.println("请假天数太多,没有人批准该假条!");
- }
- }
- }
- }
- //具体处理者3:院长类
- class Dean extends Leader {
- public void handleRequest(int LeaveDays) {
- if (LeaveDays <= 10) {
- System.out.println("院长批准您请假" + LeaveDays + "天。");
- } else {
- if (getNext() != null) {
- getNext().handleRequest(LeaveDays);
- } else {
- System.out.println("请假天数太多,没有人批准该假条!");
- }
- }
- }
- }
- //具体处理者4:教务处长类
- class DeanOfStudies extends Leader {
- public void handleRequest(int LeaveDays) {
- if (LeaveDays <= 20) {
- System.out.println("教务处长批准您请假" + LeaveDays + "天。");
- } else {
- if (getNext() != null) {
- getNext().handleRequest(LeaveDays);
- } else {
- System.out.println("请假天数太多,没有人批准该假条!");
- }
- }
- }
- }
程序运行结果如下:
院长批准您请假8天。
假如增加一个教务处长类,可以批准学生请假 20 天,也非常简单,代码如下:
- //具体处理者4:教务处长类
- class DeanOfStudies extends Leader {
- public void handleRequest(int LeaveDays) {
- if (LeaveDays <= 20) {
- System.out.println("教务处长批准您请假" + LeaveDays + "天。");
- } else {
- if (getNext() != null) {
- getNext().handleRequest(LeaveDays);
- } else {
- System.out.println("请假天数太多,没有人批准该假条!");
- }
- }
- }
- }
前边已经讲述了关于责任链模式的结构与特点,下面介绍其应用场景,责任链模式通常在以下几种情况使用。
职责链模式存在以下两种情况。
在软件开发过程中,应用程序中的部分对象可能会根据不同的情况做出不同的行为,我们把这种对象称为有状态的对象,而把影响对象行为的一个或多个动态变化的属性称为状态。当有状态的对象与外部事件产生互动时,其内部状态就会发生改变,从而使其行为也发生改变。如人都有高兴和伤心的时候,不同的情绪有不同的行为,当然外界也会影响其情绪变化。
对这种有状态的对象编程,传统的解决方案是:将这些所有可能发生的情况全都考虑到,然后使用 if-else 或 switch-case 语句来做状态判断,再进行不同情况的处理。但是显然这种做法对复杂的状态判断存在天然弊端,条件判断语句会过于臃肿,可读性差,且不具备扩展性,维护难度也大。且增加新的状态时要添加新的 if-else 语句,这违背了“开闭原则”,不利于程序的扩展。
以上问题如果采用“状态模式”就能很好地得到解决。状态模式的解决思想是:当控制一个对象状态转换的条件表达式过于复杂时,把相关“判断逻辑”提取出来,用各个不同的类进行表示,系统处于哪种情况,直接使用相应的状态类对象进行处理,这样能把原来复杂的逻辑判断简单化,消除了 if-else、switch-case 等冗余语句,代码更有层次性,并且具备良好的扩展力。
状态(State)模式的定义:对有状态的对象,把复杂的“判断逻辑”提取到不同的状态对象中,允许状态对象在其内部状态发生改变时改变其行为。
状态模式是一种对象行为型模式,其主要优点如下。
状态模式的主要缺点如下。
状态模式把受环境改变的对象行为包装在不同的状态对象里,其意图是让一个对象在其内部状态改变的时候,其行为也随之改变。现在我们来分析其基本结构和实现方法。
1. 模式的结构
状态模式包含以下主要角色。
其结构图如图 1 所示。
图1 状态模式的结构图
2. 模式的实现
状态模式的实现代码如下:
- public class StatePatternClient {
- public static void main(String[] args) {
- Context context = new Context(); //创建环境
- context.Handle(); //处理请求
- context.Handle();
- context.Handle();
- context.Handle();
- }
- }
- //环境类
- class Context {
- private State state;
- //定义环境类的初始状态
- public Context() {
- this.state = new ConcreteStateA();
- }
- //设置新状态
- public void setState(State state) {
- this.state = state;
- }
- //读取状态
- public State getState() {
- return (state);
- }
- //对请求做处理
- public void Handle() {
- state.Handle(this);
- }
- }
- //抽象状态类
- abstract class State {
- public abstract void Handle(Context context);
- }
- //具体状态A类
- class ConcreteStateA extends State {
- public void Handle(Context context) {
- System.out.println("当前状态是 A.");
- context.setState(new ConcreteStateB());
- }
- }
- //具体状态B类
- class ConcreteStateB extends State {
- public void Handle(Context context) {
- System.out.println("当前状态是 B.");
- context.setState(new ConcreteStateA());
- }
- }
程序运行结果如下:
当前状态是 A. 当前状态是 B. 当前状态是 A. 当前状态是 B.
【例1】用“状态模式”设计一个学生成绩的状态转换程序。
分析:本实例包含了“不及格”“中等”和“优秀” 3 种状态,当学生的分数小于 60 分时为“不及格”状态,当分数大于等于 60 分且小于 90 分时为“中等”状态,当分数大于等于 90 分时为“优秀”状态,我们用状态模式来实现这个程序。
首先,定义一个抽象状态类(AbstractState),其中包含了环境属性、状态名属性和当前分数属性,以及加减分方法 addScore(intx) 和检查当前状态的抽象方法 checkState()。
然后,定义“不及格”状态类 LowState、“中等”状态类 MiddleState 和“优秀”状态类 HighState,它们是具体状态类,实现 checkState() 方法,负责检査自己的状态,并根据情况转换。
最后,定义环境类(ScoreContext),其中包含了当前状态对象和加减分的方法 add(int score),客户类通过该方法来改变成绩状态。图 2 所示是其结构图。
图2 学生成绩的状态转换程序的结构图
程序代码如下:
- public class ScoreStateTest {
- public static void main(String[] args) {
- ScoreContext account = new ScoreContext();
- System.out.println("学生成绩状态测试:");
- account.add(30);
- account.add(40);
- account.add(25);
- account.add(-15);
- account.add(-25);
- }
- }
- //环境类
- class ScoreContext {
- private AbstractState state;
- ScoreContext() {
- state = new LowState(this);
- }
- public void setState(AbstractState state) {
- this.state = state;
- }
- public AbstractState getState() {
- return state;
- }
- public void add(int score) {
- state.addScore(score);
- }
- }
- //抽象状态类
- abstract class AbstractState {
- protected ScoreContext hj; //环境
- protected String stateName; //状态名
- protected int score; //分数
- public abstract void checkState(); //检查当前状态
- public void addScore(int x) {
- score += x;
- System.out.print("加上:" + x + "分,\t当前分数:" + score);
- checkState();
- System.out.println("分,\t当前状态:" + hj.getState().stateName);
- }
- }
- //具体状态类:不及格
- class LowState extends AbstractState {
- public LowState(ScoreContext h) {
- hj = h;
- stateName = "不及格";
- score = 0;
- }
- public LowState(AbstractState state) {
- hj = state.hj;
- stateName = "不及格";
- score = state.score;
- }
- public void checkState() {
- if (score >= 90) {
- hj.setState(new HighState(this));
- } else if (score >= 60) {
- hj.setState(new MiddleState(this));
- }
- }
- }
- //具体状态类:中等
- class MiddleState extends AbstractState {
- public MiddleState(AbstractState state) {
- hj = state.hj;
- stateName = "中等";
- score = state.score;
- }
- public void checkState() {
- if (score < 60) {
- hj.setState(new LowState(this));
- } else if (score >= 90) {
- hj.setState(new HighState(this));
- }
- }
- }
- //具体状态类:优秀
- class HighState extends AbstractState {
- public HighState(AbstractState state) {
- hj = state.hj;
- stateName = "优秀";
- score = state.score;
- }
- public void checkState() {
- if (score < 60) {
- hj.setState(new LowState(this));
- } else if (score < 90) {
- hj.setState(new MiddleState(this));
- }
- }
- }
程序运行结果如下:
学生成绩状态测试: 加上:30分, 当前分数:30分, 当前状态:不及格 加上:40分, 当前分数:70分, 当前状态:中等 加上:25分, 当前分数:95分, 当前状态:优秀 加上:-15分, 当前分数:80分, 当前状态:中等 加上:-25分, 当前分数:55分, 当前状态:不及格
【例2】用“状态模式”设计一个多线程的状态转换程序。
分析:多线程存在 5 种状态,分别为新建状态、就绪状态、运行状态、阻塞状态和死亡状态,各个状态当遇到相关方法调用或事件触发时会转换到其他状态,其状态转换规律如图 3 所示。
图3 线程状态转换图
现在先定义一个抽象状态类(TheadState),然后为图 3 所示的每个状态设计一个具体状态类,它们是新建状态(New)、就绪状态(Runnable )、运行状态(Running)、阻塞状态(Blocked)和死亡状态(Dead),每个状态中有触发它们转变状态的方法,环境类(ThreadContext)中先生成一个初始状态(New),并提供相关触发方法,图 4 所示是线程状态转换程序的结构图。
图4 线程状态转换程序的结构图
程序代码如下:
- public class ScoreStateTest {
- public static void main(String[] args) {
- ThreadContext context = new ThreadContext();
- context.start();
- context.getCPU();
- context.suspend();
- context.resume();
- context.getCPU();
- context.stop();
- }
- }
- //环境类
- class ThreadContext {
- private ThreadState state;
- ThreadContext() {
- state = new New();
- }
- public void setState(ThreadState state) {
- this.state = state;
- }
- public ThreadState getState() {
- return state;
- }
- public void start() {
- ((New) state).start(this);
- }
- public void getCPU() {
- ((Runnable) state).getCPU(this);
- }
- public void suspend() {
- ((Running) state).suspend(this);
- }
- public void stop() {
- ((Running) state).stop(this);
- }
- public void resume() {
- ((Blocked) state).resume(this);
- }
- }
- //抽象状态类:线程状态
- abstract class ThreadState {
- protected String stateName; //状态名
- }
- //具体状态类:新建状态
- class New extends ThreadState {
- public New() {
- stateName = "新建状态";
- System.out.println("当前线程处于:新建状态.");
- }
- public void start(ThreadContext hj) {
- System.out.print("调用start()方法-->");
- if (stateName.equals("新建状态")) {
- hj.setState(new Runnable());
- } else {
- System.out.println("当前线程不是新建状态,不能调用start()方法.");
- }
- }
- }
- //具体状态类:就绪状态
- class Runnable extends ThreadState {
- public Runnable() {
- stateName = "就绪状态";
- System.out.println("当前线程处于:就绪状态.");
- }
- public void getCPU(ThreadContext hj) {
- System.out.print("获得CPU时间-->");
- if (stateName.equals("就绪状态")) {
- hj.setState(new Running());
- } else {
- System.out.println("当前线程不是就绪状态,不能获取CPU.");
- }
- }
- }
- //具体状态类:运行状态
- class Running extends ThreadState {
- public Running() {
- stateName = "运行状态";
- System.out.println("当前线程处于:运行状态.");
- }
- public void suspend(ThreadContext hj) {
- System.out.print("调用suspend()方法-->");
- if (stateName.equals("运行状态")) {
- hj.setState(new Blocked());
- } else {
- System.out.println("当前线程不是运行状态,不能调用suspend()方法.");
- }
- }
- public void stop(ThreadContext hj) {
- System.out.print("调用stop()方法-->");
- if (stateName.equals("运行状态")) {
- hj.setState(new Dead());
- } else {
- System.out.println("当前线程不是运行状态,不能调用stop()方法.");
- }
- }
- }
- //具体状态类:阻塞状态
- class Blocked extends ThreadState {
- public Blocked() {
- stateName = "阻塞状态";
- System.out.println("当前线程处于:阻塞状态.");
- }
- public void resume(ThreadContext hj) {
- System.out.print("调用resume()方法-->");
- if (stateName.equals("阻塞状态")) {
- hj.setState(new Runnable());
- } else {
- System.out.println("当前线程不是阻塞状态,不能调用resume()方法.");
- }
- }
- }
- //具体状态类:死亡状态
- class Dead extends ThreadState {
- public Dead() {
- stateName = "死亡状态";
- System.out.println("当前线程处于:死亡状态.");
- }
- }
程序运行结果如下:
当前线程处于:新建状态. 调用start()方法-->当前线程处于:就绪状态. 获得CPU时间-->当前线程处于:运行状态. 调用suspend()方法-->当前线程处于:阻塞状态. 调用resume()方法-->当前线程处于:就绪状态. 获得CPU时间-->当前线程处于:运行状态. 调用stop()方法-->当前线程处于:死亡状态.
通常在以下情况下可以考虑使用状态模式。
在有些情况下,可能有多个环境对象需要共享一组状态,这时需要引入享元模式,将这些具体状态对象放在集合中供程序共享,其结构图如图 5 所示。
图5 共享状态模式的结构图
分析:共享状态模式的不同之处是在环境类中增加了一个 HashMap 来保存相关状态,当需要某种状态时可以从中获取,其程序代码如下:
- package state;
- import java.util.HashMap;
- public class FlyweightStatePattern {
- public static void main(String[] args) {
- ShareContext context = new ShareContext(); //创建环境
- context.Handle(); //处理请求
- context.Handle();
- context.Handle();
- context.Handle();
- }
- }
- //环境类
- class ShareContext {
- private ShareState state;
- private HashMap<String, ShareState> stateSet = new HashMap<String, ShareState>();
- public ShareContext() {
- state = new ConcreteState1();
- stateSet.put("1", state);
- state = new ConcreteState2();
- stateSet.put("2", state);
- state = getState("1");
- }
- //设置新状态
- public void setState(ShareState state) {
- this.state = state;
- }
- //读取状态
- public ShareState getState(String key) {
- ShareState s = (ShareState) stateSet.get(key);
- return s;
- }
- //对请求做处理
- public void Handle() {
- state.Handle(this);
- }
- }
- //抽象状态类
- abstract class ShareState {
- public abstract void Handle(ShareContext context);
- }
- //具体状态1类
- class ConcreteState1 extends ShareState {
- public void Handle(ShareContext context) {
- System.out.println("当前状态是: 状态1");
- context.setState(context.getState("2"));
- }
- }
- //具体状态2类
- class ConcreteState2 extends ShareState {
- public void Handle(ShareContext context) {
- System.out.println("当前状态是: 状态2");
- context.setState(context.getState("1"));
- }
- }
程序运行结果如下:
当前状态是: 状态1 当前状态是: 状态2 当前状态是: 状态1 当前状态是: 状态2
状态模式与责任链模式的区别
状态模式和责任链模式都能消除 if-else 分支过多的问题。但在某些情况下,状态模式中的状态可以理解为责任,那么在这种情况下,两种模式都可以使用。
从定义来看,状态模式强调的是一个对象内在状态的改变,而责任链模式强调的是外部节点对象间的改变。
从代码实现上来看,两者最大的区别就是状态模式的各个状态对象知道自己要进入的下一个状态对象,而责任链模式并不清楚其下一个节点处理对象,因为链式组装由客户端负责。
状态模式与策略模式的区别
状态模式和策略模式的 UML 类图架构几乎完全一样,但两者的应用场景是不一样的。策略模式的多种算法行为择其一都能满足,彼此之间是独立的,用户可自行更换策略算法,而状态模式的各个状态间存在相互关系,彼此之间在一定条件下存在自动切换状态的效果,并且用户无法指定状态,只能设置初始状态。
在现实世界中,许多对象并不是独立存在的,其中一个对象的行为发生改变可能会导致一个或者多个其他对象的行为也发生改变。例如,某种商品的物价上涨时会导致部分商家高兴,而消费者伤心;还有,当我们开车到交叉路口时,遇到红灯会停,遇到绿灯会行。这样的例子还有很多,例如,股票价格与股民、微信公众号与微信用户、气象局的天气预报与听众、小偷与警察等。
在软件世界也是这样,例如,Excel 中的数据与折线图、饼状图、柱状图之间的关系;MVC 模式中的模型与视图的关系;事件模型中的事件源与事件处理者。所有这些,如果用观察者模式来实现就非常方便。
观察者(Observer)模式的定义:指多个对象间存在一对多的依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都得到通知并被自动更新。这种模式有时又称作发布-订阅模式、模型-视图模式,它是对象行为型模式。
观察者模式是一种对象行为型模式,其主要优点如下。
它的主要缺点如下。
实现观察者模式时要注意具体目标对象和具体观察者对象之间不能直接调用,否则将使两者之间紧密耦合起来,这违反了面向对象的设计原则。
1. 模式的结构
观察者模式的主要角色如下。
观察者模式的结构图如图 1 所示。
图1 观察者模式的结构图
2. 模式的实现
观察者模式的实现代码如下:
- package net.biancheng.c.observer;
- import java.util.*;
- public class ObserverPattern {
- public static void main(String[] args) {
- Subject subject = new ConcreteSubject();
- Observer obs1 = new ConcreteObserver1();
- Observer obs2 = new ConcreteObserver2();
- subject.add(obs1);
- subject.add(obs2);
- subject.notifyObserver();
- }
- }
- //抽象目标
- abstract class Subject {
- protected List<Observer> observers = new ArrayList<Observer>();
- //增加观察者方法
- public void add(Observer observer) {
- observers.add(observer);
- }
- //删除观察者方法
- public void remove(Observer observer) {
- observers.remove(observer);
- }
- public abstract void notifyObserver(); //通知观察者方法
- }
- //具体目标
- class ConcreteSubject extends Subject {
- public void notifyObserver() {
- System.out.println("具体目标发生改变...");
- System.out.println("--------------");
- for (Object obs : observers) {
- ((Observer) obs).response();
- }
- }
- }
- //抽象观察者
- interface Observer {
- void response(); //反应
- }
- //具体观察者1
- class ConcreteObserver1 implements Observer {
- public void response() {
- System.out.println("具体观察者1作出反应!");
- }
- }
- //具体观察者1
- class ConcreteObserver2 implements Observer {
- public void response() {
- System.out.println("具体观察者2作出反应!");
- }
- }
程序运行结果如下:
具体目标发生改变... -------------- 具体观察者1作出反应! 具体观察者2作出反应!
【例1】利用观察者模式设计一个程序,分析“人民币汇率”的升值或贬值对进口公司进口产品成本或出口公司的出口产品收入以及公司利润率的影响。
分析:当“人民币汇率”升值时,进口公司的进口产品成本降低且利润率提升,出口公司的出口产品收入降低且利润率降低;当“人民币汇率”贬值时,进口公司的进口产品成本提升且利润率降低,出口公司的出口产品收入提升且利润率提升。
这里的汇率(Rate)类是抽象目标类,它包含了保存观察者(Company)的 List 和增加/删除观察者的方法,以及有关汇率改变的抽象方法 change(int number);而人民币汇率(RMBrate)类是具体目标, 它实现了父类的 change(int number) 方法,即当人民币汇率发生改变时通过相关公司;公司(Company)类是抽象观察者,它定义了一个有关汇率反应的抽象方法 response(int number);进口公司(ImportCompany)类和出口公司(ExportCompany)类是具体观察者类,它们实现了父类的 response(int number) 方法,即当它们接收到汇率发生改变的通知时作为相应的反应。图 2 所示是其结构图。
图2 人民币汇率分析程序的结构图
程序代码如下:
- package net.biancheng.c.observer;
- import java.util.*;
- public class RMBrateTest {
- public static void main(String[] args) {
- Rate rate = new RMBrate();
- Company watcher1 = new ImportCompany();
- Company watcher2 = new ExportCompany();
- rate.add(watcher1);
- rate.add(watcher2);
- rate.change(10);
- rate.change(-9);
- }
- }
- //抽象目标:汇率
- abstract class Rate {
- protected List<Company> companys = new ArrayList<Company>();
- //增加观察者方法
- public void add(Company company) {
- companys.add(company);
- }
- //删除观察者方法
- public void remove(Company company) {
- companys.remove(company);
- }
- public abstract void change(int number);
- }
- //具体目标:人民币汇率
- class RMBrate extends Rate {
- public void change(int number) {
- for (Company obs : companys) {
- ((Company) obs).response(number);
- }
- }
- }
- //抽象观察者:公司
- interface Company {
- void response(int number);
- }
- //具体观察者1:进口公司
- class ImportCompany implements Company {
- public void response(int number) {
- if (number > 0) {
- System.out.println("人民币汇率升值" + number + "个基点,降低了进口产品成本,提升了进口公司利润率。");
- } else if (number < 0) {
- System.out.println("人民币汇率贬值" + (-number) + "个基点,提升了进口产品成本,降低了进口公司利润率。");
- }
- }
- }
- //具体观察者2:出口公司
- class ExportCompany implements Company {
- public void response(int number) {
- if (number > 0) {
- System.out.println("人民币汇率升值" + number + "个基点,降低了出口产品收入,降低了出口公司的销售利润率。");
- } else if (number < 0) {
- System.out.println("人民币汇率贬值" + (-number) + "个基点,提升了出口产品收入,提升了出口公司的销售利润率。");
- }
- }
- }
程序运行结果如下:
人民币汇率升值10个基点,降低了进口产品成本,提升了进口公司利润率。 人民币汇率升值10个基点,降低了出口产品收入,降低了出口公司的销售利润率。 人民币汇率贬值9个基点,提升了进口产品成本,降低了进口公司利润率。 人民币汇率贬值9个基点,提升了出口产品收入,提升了出口公司的销售利润率。
观察者模式在软件幵发中用得最多的是窗体程序设计中的事件处理,窗体中的所有组件都是“事件源”,也就是目标对象,而事件处理程序类的对象是具体观察者对象。下面以一个学校铃声的事件处理程序为例,介绍 Windows 中的“事件处理模型”的工作原理。
【例2】利用观察者模式设计一个学校铃声的事件处理程序。
分析:在本实例中,学校的“铃”是事件源和目标,“老师”和“学生”是事件监听器和具体观察者,“铃声”是事件类。学生和老师来到学校的教学区,都会注意学校的铃,这叫事件绑定;当上课时间或下课时间到,会触发铃发声,这时会生成“铃声”事件;学生和老师听到铃声会开始上课或下课,这叫事件处理。这个实例非常适合用观察者模式实现,图 3 给出了学校铃声的事件模型。
图3 学校铃声的事件模型图
现在用“观察者模式”来实现该事件处理模型。
首先,定义一个铃声事件(RingEvent)类,它记录了铃声的类型(上课铃声/下课铃声)。
再定义一个学校的铃(BellEventSource)类,它是事件源,是观察者目标类,该类里面包含了监听器容器 listener,可以绑定监听者(学生或老师),并且有产生铃声事件和通知所有监听者的方法。
然后,定义铃声事件监听者(BellEventListener)类,它是抽象观察者,它包含了铃声事件处理方法 heardBell(RingEvent e)。
最后,定义老师类(TeachEventListener)和学生类(StuEventListener),它们是事件监听器,是具体观察者,听到铃声会去上课或下课。图 4 给出了学校铃声事件处理程序的结构。
图4 学校铃声事件处理程序的结构图
程序代码如下:
- package net.biancheng.c.observer;
- import java.util.*;
- public class BellEventTest {
- public static void main(String[] args) {
- BellEventSource bell = new BellEventSource(); //铃(事件源)
- bell.addPersonListener(new TeachEventListener()); //注册监听器(老师)
- bell.addPersonListener(new StuEventListener()); //注册监听器(学生)
- bell.ring(true); //打上课铃声
- System.out.println("------------");
- bell.ring(false); //打下课铃声
- }
- }
- //铃声事件类:用于封装事件源及一些与事件相关的参数
- class RingEvent extends EventObject {
- private static final long serialVersionUID = 1L;
- private boolean sound; //true表示上课铃声,false表示下课铃声
- public RingEvent(Object source, boolean sound) {
- super(source);
- this.sound = sound;
- }
- public void setSound(boolean sound) {
- this.sound = sound;
- }
- public boolean getSound() {
- return this.sound;
- }
- }
- //目标类:事件源,铃
- class BellEventSource {
- private List<BellEventListener> listener; //监听器容器
- public BellEventSource() {
- listener = new ArrayList<BellEventListener>();
- }
- //给事件源绑定监听器
- public void addPersonListener(BellEventListener ren) {
- listener.add(ren);
- }
- //事件触发器:敲钟,当铃声sound的值发生变化时,触发事件。
- public void ring(boolean sound) {
- String type = sound ? "上课铃" : "下课铃";
- System.out.println(type + "响!");
- RingEvent event = new RingEvent(this, sound);
- notifies(event); //通知注册在该事件源上的所有监听器
- }
- //当事件发生时,通知绑定在该事件源上的所有监听器做出反应(调用事件处理方法)
- protected void notifies(RingEvent e) {
- BellEventListener ren = null;
- Iterator<BellEventListener> iterator = listener.iterator();
- while (iterator.hasNext()) {
- ren = iterator.next();
- ren.heardBell(e);
- }
- }
- }
- //抽象观察者类:铃声事件监听器
- interface BellEventListener extends EventListener {
- //事件处理方法,听到铃声
- public void heardBell(RingEvent e);
- }
- //具体观察者类:老师事件监听器
- class TeachEventListener implements BellEventListener {
- public void heardBell(RingEvent e) {
- if (e.getSound()) {
- System.out.println("老师上课了...");
- } else {
- System.out.println("老师下课了...");
- }
- }
- }
- //具体观察者类:学生事件监听器
- class StuEventListener implements BellEventListener {
- public void heardBell(RingEvent e) {
- if (e.getSound()) {
- System.out.println("同学们,上课了...");
- } else {
- System.out.println("同学们,下课了...");
- }
- }
- }
程序运行结果如下:
上课铃响! 老师上课了... 同学们,上课了... ------------ 下课铃响! 老师下课了... 同学们,下课了...
在软件系统中,当系统一方行为依赖另一方行为的变动时,可使用观察者模式松耦合联动双方,使得一方的变动可以通知到感兴趣的另一方对象,从而让另一方对象对此做出响应。
通过前面的分析与应用实例可知观察者模式适合以下几种情形。
在 Java 中,通过 java.util.Observable 类和 java.util.Observer 接口定义了观察者模式,只要实现它们的子类就可以编写观察者模式实例。
1. Observable类
Observable 类是抽象目标类,它有一个 Vector 向量,用于保存所有要通知的观察者对象,下面来介绍它最重要的 3 个方法。
2. Observer 接口
Observer 接口是抽象观察者,它监视目标对象的变化,当目标对象发生变化时,观察者得到通知,并调用 void update(Observable o,Object arg) 方法,进行相应的工作。
【例3】利用 Observable 类和 Observer 接口实现原油期货的观察者模式实例。
分析:当原油价格上涨时,空方伤心,多方局兴;当油价下跌时,空方局兴,多方伤心。本实例中的抽象目标(Observable)类在 Java 中已经定义,可以直接定义其子类,即原油期货(OilFutures)类,它是具体目标类,该类中定义一个 SetPriCe(float price) 方法,当原油数据发生变化时调用其父类的 notifyObservers(Object arg) 方法来通知所有观察者;另外,本实例中的抽象观察者接口(Observer)在 Java 中已经定义,只要定义其子类,即具体观察者类(包括多方类 Bull 和空方类 Bear),并实现 update(Observable o,Object arg) 方法即可。图 5 所示是其结构图。
图5 原油期货的观察者模式实例的结构图
程序代码如下:
- package net.biancheng.c.observer;
- import java.util.Observer;
- import java.util.Observable;
- public class CrudeOilFutures {
- public static void main(String[] args) {
- OilFutures oil = new OilFutures();
- Observer bull = new Bull(); //多方
- Observer bear = new Bear(); //空方
- oil.addObserver(bull);
- oil.addObserver(bear);
- oil.setPrice(10);
- oil.setPrice(-8);
- }
- }
- //具体目标类:原油期货
- class OilFutures extends Observable {
- private float price;
- public float getPrice() {
- return this.price;
- }
- public void setPrice(float price) {
- super.setChanged(); //设置内部标志位,注明数据发生变化
- super.notifyObservers(price); //通知观察者价格改变了
- this.price = price;
- }
- }
- //具体观察者类:多方
- class Bull implements Observer {
- public void update(Observable o, Object arg) {
- Float price = ((Float) arg).floatValue();
- if (price > 0) {
- System.out.println("油价上涨" + price + "元,多方高兴了!");
- } else {
- System.out.println("油价下跌" + (-price) + "元,多方伤心了!");
- }
- }
- }
- //具体观察者类:空方
- class Bear implements Observer {
- public void update(Observable o, Object arg) {
- Float price = ((Float) arg).floatValue();
- if (price > 0) {
- System.out.println("油价上涨" + price + "元,空方伤心了!");
- } else {
- System.out.println("油价下跌" + (-price) + "元,空方高兴了!");
- }
- }
- }
程序运行结果如下:
油价上涨10.0元,空方伤心了! 油价上涨10.0元,多方高兴了! 油价下跌8.0元,空方高兴了! 油价下跌8.0元,多方伤心了!
在现实生活中,常常会出现好多对象之间存在复杂的交互关系,这种交互关系常常是“网状结构”,它要求每个对象都必须知道它需要交互的对象。例如,每个人必须记住他(她)所有朋友的电话;而且,朋友中如果有人的电话修改了,他(她)必须让其他所有的朋友一起修改,这叫作“牵一发而动全身”,非常复杂。
如果把这种“网状结构”改为“星形结构”的话,将大大降低它们之间的“耦合性”,这时只要找一个“中介者”就可以了。如前面所说的“每个人必须记住所有朋友电话”的问题,只要在网上建立一个每个朋友都可以访问的“通信录”就解决了。这样的例子还有很多,例如,你刚刚参加工作想租房,可以找“房屋中介”;或者,自己刚刚到一个陌生城市找工作,可以找“人才交流中心”帮忙。
在软件的开发过程中,这样的例子也很多,例如,在 MVC 框架中,控制器(C)就是模型(M)和视图(V)的中介者;还有大家常用的 QQ 聊天程序的“中介者”是 QQ 服务器。所有这些,都可以采用“中介者模式”来实现,它将大大降低对象之间的耦合性,提高系统的灵活性。
中介者(Mediator)模式的定义:定义一个中介对象来封装一系列对象之间的交互,使原有对象之间的耦合松散,且可以独立地改变它们之间的交互。中介者模式又叫调停模式,它是迪米特法则的典型应用。
中介者模式是一种对象行为型模式,其主要优点如下。
其主要缺点是:中介者模式将原本多个对象直接的相互依赖变成了中介者和多个同事类的依赖关系。当同事类越多时,中介者就会越臃肿,变得复杂且难以维护。
中介者模式实现的关键是找出“中介者”,下面对它的结构和实现进行分析。
1. 模式的结构
中介者模式包含以下主要角色。
中介者模式的结构图如图 1 所示。
图1 中介者模式的结构图
2. 模式的实现
中介者模式的实现代码如下:
- package net.biancheng.c.mediator;
- import java.util.*;
- public class MediatorPattern {
- public static void main(String[] args) {
- Mediator md = new ConcreteMediator();
- Colleague c1, c2;
- c1 = new ConcreteColleague1();
- c2 = new ConcreteColleague2();
- md.register(c1);
- md.register(c2);
- c1.send();
- System.out.println("-------------");
- c2.send();
- }
- }
- //抽象中介者
- abstract class Mediator {
- public abstract void register(Colleague colleague);
- public abstract void relay(Colleague cl); //转发
- }
- //具体中介者
- class ConcreteMediator extends Mediator {
- private List<Colleague> colleagues = new ArrayList<Colleague>();
- public void register(Colleague colleague) {
- if (!colleagues.contains(colleague)) {
- colleagues.add(colleague);
- colleague.setMedium(this);
- }
- }
- public void relay(Colleague cl) {
- for (Colleague ob : colleagues) {
- if (!ob.equals(cl)) {
- ((Colleague) ob).receive();
- }
- }
- }
- }
- //抽象同事类
- abstract class Colleague {
- protected Mediator mediator;
- public void setMedium(Mediator mediator) {
- this.mediator = mediator;
- }
- public abstract void receive();
- public abstract void send();
- }
- //具体同事类
- class ConcreteColleague1 extends Colleague {
- public void receive() {
- System.out.println("具体同事类1收到请求。");
- }
- public void send() {
- System.out.println("具体同事类1发出请求。");
- mediator.relay(this); //请中介者转发
- }
- }
- //具体同事类
- class ConcreteColleague2 extends Colleague {
- public void receive() {
- System.out.println("具体同事类2收到请求。");
- }
- public void send() {
- System.out.println("具体同事类2发出请求。");
- mediator.relay(this); //请中介者转发
- }
- }
程序的运行结果如下:
具体同事类1发出请求。 具体同事类2收到请求。 ------------- 具体同事类2发出请求。 具体同事类1收到请求。
【例1】用中介者模式编写一个“韶关房地产交流平台”程序。
说明:韶关房地产交流平台是“房地产中介公司”提供给“卖方客户”与“买方客户”进行信息交流的平台,比较适合用中介者模式来实现。
首先,定义一个中介公司(Medium)接口,它是抽象中介者,它包含了客户注册方法 register(Customer member) 和信息转发方法 relay(String from,String ad);再定义一个韶关房地产中介(EstateMedium)公司,它是具体中介者类,它包含了保存客户信息的 List 对象,并实现了中介公司中的抽象方法。
然后,定义一个客户(Customer)类,它是抽象同事类,其中包含了中介者的对象,和发送信息的 send(String ad) 方法与接收信息的 receive(String from,String ad) 方法的接口,由于本程序是窗体程序,所以本类继承 JPmme 类,并实现动作事件的处理方法 actionPerformed(ActionEvent e)。
最后,定义卖方(Seller)类和买方(Buyer)类,它们是具体同事类,是客户(Customer)类的子类,它们实现了父类中的抽象方法,通过中介者类进行信息交流,其结构图如图 2 所示。
图2 韶关房地产交流平台的结构图
程序代码如下:
- package net.biancheng.c.mediator;
- import javax.swing.*;
- import java.awt.*;
- import java.awt.event.ActionEvent;
- import java.awt.event.ActionListener;
- import java.util.ArrayList;
- import java.util.List;
- public class DatingPlatform {
- public static void main(String[] args) {
- Medium md = new EstateMedium(); //房产中介
- Customer member1, member2;
- member1 = new Seller("张三(卖方)");
- member2 = new Buyer("李四(买方)");
- md.register(member1); //客户注册
- md.register(member2);
- }
- }
- //抽象中介者:中介公司
- interface Medium {
- void register(Customer member); //客户注册
- void relay(String from, String ad); //转发
- }
- //具体中介者:房地产中介
- class EstateMedium implements Medium {
- private List<Customer> members = new ArrayList<Customer>();
- public void register(Customer member) {
- if (!members.contains(member)) {
- members.add(member);
- member.setMedium(this);
- }
- }
- public void relay(String from, String ad) {
- for (Customer ob : members) {
- String name = ob.getName();
- if (!name.equals(from)) {
- ((Customer) ob).receive(from, ad);
- }
- }
- }
- }
- //抽象同事类:客户
- abstract class Customer extends JFrame implements ActionListener {
- private static final long serialVersionUID = -7219939540794786080L;
- protected Medium medium;
- protected String name;
- JTextField SentText;
- JTextArea ReceiveArea;
- public Customer(String name) {
- super(name);
- this.name = name;
- }
- void ClientWindow(int x, int y) {
- Container cp;
- JScrollPane sp;
- JPanel p1, p2;
- cp = this.getContentPane();
- SentText = new JTextField(18);
- ReceiveArea = new JTextArea(10, 18);
- ReceiveArea.setEditable(false);
- p1 = new JPanel();
- p1.setBorder(BorderFactory.createTitledBorder("接收内容:"));
- p1.add(ReceiveArea);
- sp = new JScrollPane(p1);
- cp.add(sp, BorderLayout.NORTH);
- p2 = new JPanel();
- p2.setBorder(BorderFactory.createTitledBorder("发送内容:"));
- p2.add(SentText);
- cp.add(p2, BorderLayout.SOUTH);
- SentText.addActionListener(this);
- this.setLocation(x, y);
- this.setSize(250, 330);
- this.setResizable(false); //窗口大小不可调整
- this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
- this.setVisible(true);
- }
- public void actionPerformed(ActionEvent e) {
- String tempInfo = SentText.getText().trim();
- SentText.setText("");
- this.send(tempInfo);
- }
- public String getName() {
- return name;
- }
- public void setMedium(Medium medium) {
- this.medium = medium;
- }
- public abstract void send(String ad);
- public abstract void receive(String from, String ad);
- }
- //具体同事类:卖方
- class Seller extends Customer {
- private static final long serialVersionUID = -1443076716629516027L;
- public Seller(String name) {
- super(name);
- ClientWindow(50, 100);
- }
- public void send(String ad) {
- ReceiveArea.append("我(卖方)说: " + ad + "\n");
- //使滚动条滚动到最底端
- ReceiveArea.setCaretPosition(ReceiveArea.getText().length());
- medium.relay(name, ad);
- }
- public void receive(String from, String ad) {
- ReceiveArea.append(from + "说: " + ad + "\n");
- //使滚动条滚动到最底端
- ReceiveArea.setCaretPosition(ReceiveArea.getText().length());
- }
- }
- //具体同事类:买方
- class Buyer extends Customer {
- private static final long serialVersionUID = -474879276076308825L;
- public Buyer(String name) {
- super(name);
- ClientWindow(350, 100);
- }
- public void send(String ad) {
- ReceiveArea.append("我(买方)说: " + ad + "\n");
- //使滚动条滚动到最底端
- ReceiveArea.setCaretPosition(ReceiveArea.getText().length());
- medium.relay(name, ad);
- }
- public void receive(String from, String ad) {
- ReceiveArea.append(from + "说: " + ad + "\n");
- //使滚动条滚动到最底端
- ReceiveArea.setCaretPosition(ReceiveArea.getText().length());
- }
- }
程序的运行结果如图 3 所示。
图3 韶关房地产交流平台的运行结果
前面分析了中介者模式的结构与特点,下面分析其以下应用场景。
在实际开发中,通常采用以下两种方法来简化中介者模式,使开发变得更简单。
图 4 所示是简化中介者模式的结构图。
图4 简化中介者模式的结构图
程序代码如下:
- package net.biancheng.c.mediator;
- import java.util.*;
- public class SimpleMediatorPattern {
- public static void main(String[] args) {
- SimpleColleague c1, c2;
- c1 = new SimpleConcreteColleague1();
- c2 = new SimpleConcreteColleague2();
- c1.send();
- System.out.println("-----------------");
- c2.send();
- }
- }
- //简单单例中介者
- class SimpleMediator {
- private static SimpleMediator smd = new SimpleMediator();
- private List<SimpleColleague> colleagues = new ArrayList<SimpleColleague>();
- private SimpleMediator() {
- }
- public static SimpleMediator getMedium() {
- return (smd);
- }
- public void register(SimpleColleague colleague) {
- if (!colleagues.contains(colleague)) {
- colleagues.add(colleague);
- }
- }
- public void relay(SimpleColleague scl) {
- for (SimpleColleague ob : colleagues) {
- if (!ob.equals(scl)) {
- ((SimpleColleague) ob).receive();
- }
- }
- }
- }
- //抽象同事类
- interface SimpleColleague {
- void receive();
- void send();
- }
- //具体同事类
- class SimpleConcreteColleague1 implements SimpleColleague {
- SimpleConcreteColleague1() {
- SimpleMediator smd = SimpleMediator.getMedium();
- smd.register(this);
- }
- public void receive() {
- System.out.println("具体同事类1:收到请求。");
- }
- public void send() {
- SimpleMediator smd = SimpleMediator.getMedium();
- System.out.println("具体同事类1:发出请求...");
- smd.relay(this); //请中介者转发
- }
- }
- //具体同事类
- class SimpleConcreteColleague2 implements SimpleColleague {
- SimpleConcreteColleague2() {
- SimpleMediator smd = SimpleMediator.getMedium();
- smd.register(this);
- }
- public void receive() {
- System.out.println("具体同事类2:收到请求。");
- }
- public void send() {
- SimpleMediator smd = SimpleMediator.getMedium();
- System.out.println("具体同事类2:发出请求...");
- smd.relay(this); //请中介者转发
- }
- }
程序运行结果如下:
具体同事类1:发出请求... 具体同事类2:收到请求。 ----------------- 具体同事类2:发出请求... 具体同事类1:收到请求。
在现实生活以及程序设计中,经常要访问一个聚合对象中的各个元素,如“数据结构”中的链表遍历,通常的做法是将链表的创建和遍历都放在同一个类中,但这种方式不利于程序的扩展,如果要更换遍历方法就必须修改程序源代码,这违背了 “开闭原则”。
既然将遍历方法封装在聚合类中不可取,那么聚合类中不提供遍历方法,将遍历方法由用户自己实现是否可行呢?答案是同样不可取,因为这种方式会存在两个缺点:
“迭代器模式”能较好地克服以上缺点,它在客户访问类与聚合类之间插入一个迭代器,这分离了聚合对象与其遍历行为,对客户也隐藏了其内部细节,且满足“单一职责原则”和“开闭原则”,如 Java 中的 Collection、List、Set、Map 等都包含了迭代器。
迭代器模式在生活中应用的比较广泛,比如:物流系统中的传送带,不管传送的是什么物品,都会被打包成一个个箱子,并且有一个统一的二维码。这样我们不需要关心箱子里是什么,在分发时只需要一个个检查发送的目的地即可。再比如,我们平时乘坐交通工具,都是统一刷卡或者刷脸进站,而不需要关心是男性还是女性、是残疾人还是正常人等信息。
迭代器(Iterator)模式的定义:提供一个对象来顺序访问聚合对象中的一系列数据,而不暴露聚合对象的内部表示。迭代器模式是一种对象行为型模式,其主要优点如下。
其主要缺点是:增加了类的个数,这在一定程度上增加了系统的复杂性。
在日常开发中,我们几乎不会自己写迭代器。除非需要定制一个自己实现的数据结构对应的迭代器,否则,开源框架提供的 API 完全够用。
迭代器模式是通过将聚合对象的遍历行为分离出来,抽象成迭代器类来实现的,其目的是在不暴露聚合对象的内部结构的情况下,让外部代码透明地访问聚合的内部数据。现在我们来分析其基本结构与实现方法。
1. 模式的结构
迭代器模式主要包含以下角色。
其结构图如图 1 所示。
图1 迭代器模式的结构图
2. 模式的实现
迭代器模式的实现代码如下:
- package net.biancheng.c.iterator;
- import java.util.*;
- public class IteratorPattern {
- public static void main(String[] args) {
- Aggregate ag = new ConcreteAggregate();
- ag.add("中山大学");
- ag.add("华南理工");
- ag.add("韶关学院");
- System.out.print("聚合的内容有:");
- Iterator it = ag.getIterator();
- while (it.hasNext()) {
- Object ob = it.next();
- System.out.print(ob.toString() + "\t");
- }
- Object ob = it.first();
- System.out.println("\nFirst:" + ob.toString());
- }
- }
- //抽象聚合
- interface Aggregate {
- public void add(Object obj);
- public void remove(Object obj);
- public Iterator getIterator();
- }
- //具体聚合
- class ConcreteAggregate implements Aggregate {
- private List<Object> list = new ArrayList<Object>();
- public void add(Object obj) {
- list.add(obj);
- }
- public void remove(Object obj) {
- list.remove(obj);
- }
- public Iterator getIterator() {
- return (new ConcreteIterator(list));
- }
- }
- //抽象迭代器
- interface Iterator {
- Object first();
- Object next();
- boolean hasNext();
- }
- //具体迭代器
- class ConcreteIterator implements Iterator {
- private List<Object> list = null;
- private int index = -1;
- public ConcreteIterator(List<Object> list) {
- this.list = list;
- }
- public boolean hasNext() {
- if (index < list.size() - 1) {
- return true;
- } else {
- return false;
- }
- }
- public Object first() {
- index = 0;
- Object obj = list.get(index);
- ;
- return obj;
- }
- public Object next() {
- Object obj = null;
- if (this.hasNext()) {
- obj = list.get(++index);
- }
- return obj;
- }
- }
程序运行结果如下:
聚合的内容有:中山大学 华南理工 韶关学院 First:中山大学
【例1】用迭代器模式编写一个浏览婺源旅游风景图的程序。
分析:婺源的名胜古迹较多,要设计一个查看相关景点图片(点此下载本实例所要显示的景点图片)和简介的程序,用“迭代器模式”设计比较合适。
首先,设计一个婺源景点(WyViewSpot)类来保存每张图片的名称与简介;再设计一个景点集(ViewSpotSet)接口,它是抽象聚合类,提供了增加和删除婺源景点的方法,以及获取迭代器的方法。
然后,定义一个婺源景点集(WyViewSpotSet)类,它是具体聚合类,用 ArrayList 来保存所有景点信息,并实现父类中的抽象方法;再定义婺源景点的抽象迭代器(ViewSpotltemtor)接口,其中包含了查看景点信息的相关方法。
最后,定义婺源景点的具体迭代器(WyViewSpotlterator)类,它实现了父类的抽象方法;客户端程序设计成窗口程序,它初始化婺源景点集(ViewSpotSet)中的数据,并实现 ActionListener 接口,它通过婺源景点迭代器(ViewSpotlterator)来査看婺源景点(WyViewSpot)的信息。图 2 所示是其结构图。
图2 婺源旅游风景图浏览程序的结构图(点此查看原图)
程序代码如下:
- package net.biancheng.c.iterator;
- import javax.swing.*;
- import java.awt.*;
- import java.awt.event.ActionEvent;
- import java.awt.event.ActionListener;
- import java.util.ArrayList;
- public class PictureIterator {
- public static void main(String[] args) {
- new PictureFrame();
- }
- }
- //相框类
- class PictureFrame extends JFrame implements ActionListener {
- private static final long serialVersionUID = 1L;
- ViewSpotSet ag; //婺源景点集接口
- ViewSpotIterator it; //婺源景点迭代器接口
- WyViewSpot ob; //婺源景点类
- PictureFrame() {
- super("中国最美乡村“婺源”的部分风景图");
- this.setResizable(false);
- ag = new WyViewSpotSet();
- ag.add(new WyViewSpot("江湾", "江湾景区是婺源的一个国家5A级旅游景区,景区内有萧江宗祠、永思街、滕家老屋、婺源人家、乡贤园、百工坊等一大批古建筑,精美绝伦,做工精细。"));
- ag.add(new WyViewSpot("李坑", "李坑村是一个以李姓聚居为主的古村落,是国家4A级旅游景区,其建筑风格独特,是著名的徽派建筑,给人一种安静、祥和的感觉。"));
- ag.add(new WyViewSpot("思溪延村", "思溪延村位于婺源县思口镇境内,始建于南宋庆元五年(1199年),当时建村者俞氏以(鱼)思清溪水而名。"));
- ag.add(new WyViewSpot("晓起村", "晓起有“中国茶文化第一村”与“国家级生态示范村”之美誉,村屋多为清代建筑,风格各具特色,村中小巷均铺青石,曲曲折折,回环如棋局。"));
- ag.add(new WyViewSpot("菊径村", "菊径村形状为山环水绕型,小河成大半圆型,绕村庄将近一周,四周为高山环绕,符合中国的八卦“后山前水”设计,当地人称“脸盆村”。"));
- ag.add(new WyViewSpot("篁岭", "篁岭是著名的“晒秋”文化起源地,也是一座距今近六百历史的徽州古村;篁岭属典型山居村落,民居围绕水口呈扇形梯状错落排布。"));
- ag.add(new WyViewSpot("彩虹桥", "彩虹桥是婺源颇有特色的带顶的桥——廊桥,其不仅造型优美,而且它可在雨天里供行人歇脚,其名取自唐诗“两水夹明镜,双桥落彩虹”。"));
- ag.add(new WyViewSpot("卧龙谷", "卧龙谷是国家4A级旅游区,这里飞泉瀑流泄银吐玉、彩池幽潭碧绿清新、山峰岩石挺拔奇巧,活脱脱一幅天然泼墨山水画。"));
- it = ag.getIterator(); //获取婺源景点迭代器
- ob = it.first();
- this.showPicture(ob.getName(), ob.getIntroduce());
- }
- //显示图片
- void showPicture(String Name, String Introduce) {
- Container cp = this.getContentPane();
- JPanel picturePanel = new JPanel();
- JPanel controlPanel = new JPanel();
- String FileName = "src/iterator/Picture/" + Name + ".jpg";
- JLabel lb = new JLabel(Name, new ImageIcon(FileName), JLabel.CENTER);
- JTextArea ta = new JTextArea(Introduce);
- lb.setHorizontalTextPosition(JLabel.CENTER);
- lb.setVerticalTextPosition(JLabel.TOP);
- lb.setFont(new Font("宋体", Font.BOLD, 20));
- ta.setLineWrap(true);
- ta.setEditable(false);
- //ta.setBackground(Color.orange);
- picturePanel.setLayout(new BorderLayout(5, 5));
- picturePanel.add("Center", lb);
- picturePanel.add("South", ta);
- JButton first, last, next, previous;
- first = new JButton("第一张");
- next = new JButton("下一张");
- previous = new JButton("上一张");
- last = new JButton("最末张");
- first.addActionListener(this);
- next.addActionListener(this);
- previous.addActionListener(this);
- last.addActionListener(this);
- controlPanel.add(first);
- controlPanel.add(next);
- controlPanel.add(previous);
- controlPanel.add(last);
- cp.add("Center", picturePanel);
- cp.add("South", controlPanel);
- this.setSize(630, 550);
- this.setVisible(true);
- this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
- }
- @Override
- public void actionPerformed(ActionEvent arg0) {
- String command = arg0.getActionCommand();
- if (command.equals("第一张")) {
- ob = it.first();
- this.showPicture(ob.getName(), ob.getIntroduce());
- } else if (command.equals("下一张")) {
- ob = it.next();
- this.showPicture(ob.getName(), ob.getIntroduce());
- } else if (command.equals("上一张")) {
- ob = it.previous();
- this.showPicture(ob.getName(), ob.getIntroduce());
- } else if (command.equals("最末张")) {
- ob = it.last();
- this.showPicture(ob.getName(), ob.getIntroduce());
- }
- }
- }
- //婺源景点类
- class WyViewSpot {
- private String Name;
- private String Introduce;
- WyViewSpot(String Name, String Introduce) {
- this.Name = Name;
- this.Introduce = Introduce;
- }
- public String getName() {
- return Name;
- }
- public String getIntroduce() {
- return Introduce;
- }
- }
- //抽象聚合:婺源景点集接口
- interface ViewSpotSet {
- void add(WyViewSpot obj);
- void remove(WyViewSpot obj);
- ViewSpotIterator getIterator();
- }
- //具体聚合:婺源景点集
- class WyViewSpotSet implements ViewSpotSet {
- private ArrayList<WyViewSpot> list = new ArrayList<WyViewSpot>();
- public void add(WyViewSpot obj) {
- list.add(obj);
- }
- public void remove(WyViewSpot obj) {
- list.remove(obj);
- }
- public ViewSpotIterator getIterator() {
- return (new WyViewSpotIterator(list));
- }
- }
- //抽象迭代器:婺源景点迭代器接口
- interface ViewSpotIterator {
- boolean hasNext();
- WyViewSpot first();
- WyViewSpot next();
- WyViewSpot previous();
- WyViewSpot last();
- }
- //具体迭代器:婺源景点迭代器
- class WyViewSpotIterator implements ViewSpotIterator {
- private ArrayList<WyViewSpot> list = null;
- private int index = -1;
- WyViewSpot obj = null;
- public WyViewSpotIterator(ArrayList<WyViewSpot> list) {
- this.list = list;
- }
- public boolean hasNext() {
- if (index < list.size() - 1) {
- return true;
- } else {
- return false;
- }
- }
- public WyViewSpot first() {
- index = 0;
- obj = list.get(index);
- return obj;
- }
- public WyViewSpot next() {
- if (this.hasNext()) {
- obj = list.get(++index);
- }
- return obj;
- }
- public WyViewSpot previous() {
- if (index > 0) {
- obj = list.get(--index);
- }
- return obj;
- }
- public WyViewSpot last() {
- index = list.size() - 1;
- obj = list.get(index);
- return obj;
- }
- }
程序运行结果如图 3 所示。
图3 婺源旅游风景图浏览程序的运行结果(点此查看原图)
前面介绍了关于迭代器模式的结构与特点,下面介绍其应用场景,迭代器模式通常在以下几种情况使用。
由于聚合与迭代器的关系非常密切,所以大多数语言在实现聚合类时都提供了迭代器类,因此大数情况下使用语言中已有的聚合类的迭代器就已经够了。
迭代器模式常常与组合模式结合起来使用,在对组合模式中的容器构件进行访问时,经常将迭代器潜藏在组合模式的容器构成类中。当然,也可以构造一个外部迭代器来对容器构件进行访问,其结构图如图 4 所示。
图4 组合迭代器模式的结构图
在现实生活中,有些集合对象存在多种不同的元素,且每种元素也存在多种不同的访问者和处理方式。例如,公园中存在多个景点,也存在多个游客,不同的游客对同一个景点的评价可能不同;医院医生开的处方单中包含多种药元素,査看它的划价员和药房工作人员对它的处理方式也不同,划价员根据处方单上面的药品名和数量进行划价,药房工作人员根据处方单的内容进行抓药。
这样的例子还有很多,例如,电影或电视剧中的人物角色,不同的观众对他们的评价也不同;还有顾客在商场购物时放在“购物车”中的商品,顾客主要关心所选商品的性价比,而收银员关心的是商品的价格和数量。
这些被处理的数据元素相对稳定而访问方式多种多样的数据结构,如果用“访问者模式”来处理比较方便。访问者模式能把处理方法从数据结构中分离出来,并可以根据需要增加新的处理方法,且不用修改原来的程序代码与数据结构,这提高了程序的扩展性和灵活性。
访问者(Visitor)模式的定义:将作用于某种数据结构中的各元素的操作分离出来封装成独立的类,使其在不改变数据结构的前提下可以添加作用于这些元素的新的操作,为数据结构中的每个元素提供多种访问方式。它将对数据的操作与数据结构进行分离,是行为类模式中最复杂的一种模式。
访问者(Visitor)模式是一种对象行为型模式,其主要优点如下。
访问者(Visitor)模式的主要缺点如下。
访问者(Visitor)模式实现的关键是如何将作用于元素的操作分离出来封装成独立的类,其基本结构与实现方法如下。
1. 模式的结构
访问者模式包含以下主要角色。
其结构图如图 1 所示。
图1 访问者(Visitor)模式的结构图(点此查看原图)
2. 模式的实现
访问者模式的实现代码如下:
- package net.biancheng.c.visitor;
- import java.util.*;
- public class VisitorPattern {
- public static void main(String[] args) {
- ObjectStructure os = new ObjectStructure();
- os.add(new ConcreteElementA());
- os.add(new ConcreteElementB());
- Visitor visitor = new ConcreteVisitorA();
- os.accept(visitor);
- System.out.println("------------------------");
- visitor = new ConcreteVisitorB();
- os.accept(visitor);
- }
- }
- //抽象访问者
- interface Visitor {
- void visit(ConcreteElementA element);
- void visit(ConcreteElementB element);
- }
- //具体访问者A类
- class ConcreteVisitorA implements Visitor {
- public void visit(ConcreteElementA element) {
- System.out.println("具体访问者A访问-->" + element.operationA());
- }
- public void visit(ConcreteElementB element) {
- System.out.println("具体访问者A访问-->" + element.operationB());
- }
- }
- //具体访问者B类
- class ConcreteVisitorB implements Visitor {
- public void visit(ConcreteElementA element) {
- System.out.println("具体访问者B访问-->" + element.operationA());
- }
- public void visit(ConcreteElementB element) {
- System.out.println("具体访问者B访问-->" + element.operationB());
- }
- }
- //抽象元素类
- interface Element {
- void accept(Visitor visitor);
- }
- //具体元素A类
- class ConcreteElementA implements Element {
- public void accept(Visitor visitor) {
- visitor.visit(this);
- }
- public String operationA() {
- return "具体元素A的操作。";
- }
- }
- //具体元素B类
- class ConcreteElementB implements Element {
- public void accept(Visitor visitor) {
- visitor.visit(this);
- }
- public String operationB() {
- return "具体元素B的操作。";
- }
- }
- //对象结构角色
- class ObjectStructure {
- private List<Element> list = new ArrayList<Element>();
- public void accept(Visitor visitor) {
- Iterator<Element> i = list.iterator();
- while (i.hasNext()) {
- ((Element) i.next()).accept(visitor);
- }
- }
- public void add(Element element) {
- list.add(element);
- }
- public void remove(Element element) {
- list.remove(element);
- }
- }
程序的运行结果如下:
具体访问者A访问-->具体元素A的操作。 具体访问者A访问-->具体元素B的操作。 ------------------------ 具体访问者B访问-->具体元素A的操作。 具体访问者B访问-->具体元素B的操作。
【例1】利用“访问者(Visitor)模式”模拟艺术公司与造币公司的功能。
分析:艺术公司利用“铜”可以设计出铜像,利用“纸”可以画出图画;造币公司利用“铜”可以印出铜币,利用“纸”可以印出纸币(点此下载运行该程序后所要显示的图片)。对“铜”和“纸”这两种元素,两个公司的处理方法不同,所以该实例用访问者模式来实现比较适合。
首先,定义一个公司(Company)接口,它是抽象访问者,提供了两个根据纸(Paper)或铜(Cuprum)这两种元素创建作品的方法;再定义艺术公司(ArtCompany)类和造币公司(Mint)类,它们是具体访问者,实现了父接口的方法。
然后,定义一个材料(Material)接口,它是抽象元素,提供了 accept(Company visitor)方法来接受访问者(Company)对象访问;再定义纸(Paper)类和铜(Cuprum)类,它们是具体元素类,实现了父接口中的方法。
最后,定义一个材料集(SetMaterial)类,它是对象结构角色,拥有保存所有元素的容器 List,并提供让访问者对象遍历容器中的所有元素的 accept(Company visitor)方法;客户类设计成窗体程序,它提供材料集(SetMaterial)对象供访问者(Company)对象访问,实现了 ItemListener 接口,处理用户的事件请求。图 2 所示是其结构图。
图2 艺术公司与造币公司的结构图
程序代码如下:
- package net.biancheng.c.visitor;
- import javax.swing.*;
- import java.awt.event.ItemEvent;
- import java.awt.event.ItemListener;
- import java.util.ArrayList;
- import java.util.Iterator;
- import java.util.List;
- public class VisitorProducer {
- public static void main(String[] args) {
- new MaterialWin();
- }
- }
- //窗体类
- class MaterialWin extends JFrame implements ItemListener {
- private static final long serialVersionUID = 1L;
- JPanel CenterJP;
- SetMaterial os; //材料集对象
- Company visitor1, visitor2; //访问者对象
- String[] select;
- MaterialWin() {
- super("利用访问者模式设计艺术公司和造币公司");
- JRadioButton Art;
- JRadioButton mint;
- os = new SetMaterial();
- os.add(new Cuprum());
- os.add(new Paper());
- visitor1 = new ArtCompany();//艺术公司
- visitor2 = new Mint(); //造币公司
- this.setBounds(10, 10, 750, 350);
- this.setResizable(false);
- CenterJP = new JPanel();
- this.add("Center", CenterJP);
- JPanel SouthJP = new JPanel();
- JLabel yl = new JLabel("原材料有:铜和纸,请选择生产公司:");
- Art = new JRadioButton("艺术公司", true);
- mint = new JRadioButton("造币公司");
- Art.addItemListener(this);
- mint.addItemListener(this);
- ButtonGroup group = new ButtonGroup();
- group.add(Art);
- group.add(mint);
- SouthJP.add(yl);
- SouthJP.add(Art);
- SouthJP.add(mint);
- this.add("South", SouthJP);
- select = (os.accept(visitor1)).split(" "); //获取产品名
- showPicture(select[0], select[1]); //显示产品
- }
- //显示图片
- void showPicture(String Cuprum, String paper) {
- CenterJP.removeAll(); //清除面板内容
- CenterJP.repaint(); //刷新屏幕
- String FileName1 = "src/visitor/Picture/" + Cuprum + ".jpg";
- String FileName2 = "src/visitor/Picture/" + paper + ".jpg";
- JLabel lb = new JLabel(new ImageIcon(FileName1), JLabel.CENTER);
- JLabel rb = new JLabel(new ImageIcon(FileName2), JLabel.CENTER);
- CenterJP.add(lb);
- CenterJP.add(rb);
- this.setVisible(true);
- this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
- }
- @Override
- public void itemStateChanged(ItemEvent arg0) {
- JRadioButton jc = (JRadioButton) arg0.getSource();
- if (jc.isSelected()) {
- if (jc.getText() == "造币公司") {
- select = (os.accept(visitor2)).split(" ");
- } else {
- select = (os.accept(visitor1)).split(" ");
- }
- showPicture(select[0], select[1]); //显示选择的产品
- }
- }
- }
- //抽象访问者:公司
- interface Company {
- String create(Paper element);
- String create(Cuprum element);
- }
- //具体访问者:艺术公司
- class ArtCompany implements Company {
- public String create(Paper element) {
- return "讲学图";
- }
- public String create(Cuprum element) {
- return "朱熹铜像";
- }
- }
- //具体访问者:造币公司
- class Mint implements Company {
- public String create(Paper element) {
- return "纸币";
- }
- public String create(Cuprum element) {
- return "铜币";
- }
- }
- //抽象元素:材料
- interface Material {
- String accept(Company visitor);
- }
- //具体元素:纸
- class Paper implements Material {
- public String accept(Company visitor) {
- return (visitor.create(this));
- }
- }
- //具体元素:铜
- class Cuprum implements Material {
- public String accept(Company visitor) {
- return (visitor.create(this));
- }
- }
- //对象结构角色:材料集
- class SetMaterial {
- private List<Material> list = new ArrayList<Material>();
- public String accept(Company visitor) {
- Iterator<Material> i = list.iterator();
- String tmp = "";
- while (i.hasNext()) {
- tmp += ((Material) i.next()).accept(visitor) + " ";
- }
- return tmp; //返回某公司的作品集
- }
- public void add(Material element) {
- list.add(element);
- }
- public void remove(Material element) {
- list.remove(element);
- }
- }
程序运行结果如图 3 所示。
(a)艺术公司设计的产品
(b)造币公司生产的货币
图3 艺术公司与造币公司的运行结果
当系统中存在类型数量稳定(固定)的一类数据结构时,可以使用访问者模式方便地实现对该类型所有数据结构的不同操作,而又不会对数据产生任何副作用(脏数据)。
简而言之,就是当对集合中的不同类型数据(类型数量稳定)进行多种操作时,使用访问者模式。
通常在以下情况可以考虑使用访问者(Visitor)模式。
访问者(Visitor)模式是使用频率较高的一种设计模式,它常常同以下两种设计模式联用。
(1)与“迭代器模式”联用。因为访问者模式中的“对象结构”是一个包含元素角色的容器,当访问者遍历容器中的所有元素时,常常要用迭代器。如【例1】中的对象结构是用 List 实现的,它通过 List 对象的 Iterator() 方法获取迭代器。如果对象结构中的聚合类没有提供迭代器,也可以用迭代器模式自定义一个。
(2)访问者(Visitor)模式同“组合模式”联用。因为访问者(Visitor)模式中的“元素对象”可能是叶子对象或者是容器对象,如果元素对象包含容器对象,就必须用到组合模式,其结构图如图 4 所示。
图4 包含组合模式的访问者模式的结构图
每个人都有犯错误的时候,都希望有种“后悔药”能弥补自己的过失,让自己重新开始,但现实是残酷的。在计算机应用中,客户同样会常常犯错误,能否提供“后悔药”给他们呢?当然是可以的,而且是有必要的。这个功能由“备忘录模式”来实现。
其实很多应用软件都提供了这项功能,如 Word、记事本、Photoshop、Eclipse 等软件在编辑时按 Ctrl+Z 组合键时能撤销当前操作,使文档恢复到之前的状态;还有在 IE 中的后退键、数据库事务管理中的回滚操作、玩游戏时的中间结果存档功能、数据库与操作系统的备份操作、棋类游戏中的悔棋功能等都属于这类。
备忘录模式能记录一个对象的内部状态,当用户后悔时能撤销当前操作,使数据恢复到它原先的状态。
备忘录(Memento)模式的定义:在不破坏封装性的前提下,捕获一个对象的内部状态,并在该对象之外保存这个状态,以便以后当需要时能将该对象恢复到原先保存的状态。该模式又叫快照模式。
备忘录模式是一种对象行为型模式,其主要优点如下。
其主要缺点是:资源消耗大。如果要保存的内部状态信息过多或者特别频繁,将会占用比较大的内存资源。
备忘录模式的核心是设计备忘录类以及用于管理备忘录的管理者类,现在我们来学习其结构与实现。
1. 模式的结构
备忘录模式的主要角色如下。
备忘录模式的结构图如图 1 所示。
图1 备忘录模式的结构图
2. 模式的实现
备忘录模式的实现代码如下:
- package net.biancheng.c.memento;
- public class MementoPattern {
- public static void main(String[] args) {
- Originator or = new Originator();
- Caretaker cr = new Caretaker();
- or.setState("S0");
- System.out.println("初始状态:" + or.getState());
- cr.setMemento(or.createMemento()); //保存状态
- or.setState("S1");
- System.out.println("新的状态:" + or.getState());
- or.restoreMemento(cr.getMemento()); //恢复状态
- System.out.println("恢复状态:" + or.getState());
- }
- }
- //备忘录
- class Memento {
- private String state;
- public Memento(String state) {
- this.state = state;
- }
- public void setState(String state) {
- this.state = state;
- }
- public String getState() {
- return state;
- }
- }
- //发起人
- class Originator {
- private String state;
- public void setState(String state) {
- this.state = state;
- }
- public String getState() {
- return state;
- }
- public Memento createMemento() {
- return new Memento(state);
- }
- public void restoreMemento(Memento m) {
- this.setState(m.getState());
- }
- }
- //管理者
- class Caretaker {
- private Memento memento;
- public void setMemento(Memento m) {
- memento = m;
- }
- public Memento getMemento() {
- return memento;
- }
- }
程序运行的结果如下:
初始状态:S0 新的状态:S1 恢复状态:S0
【例1】利用备忘录模式设计相亲游戏。
分析:假如有西施、王昭君、貂蝉、杨玉环四大美女同你相亲,你可以选择其中一位作为你的爱人;当然,如果你对前面的选择不满意,还可以重新选择,但希望你不要太花心;这个游戏提供后悔功能,用“备忘录模式”设计比较合适(点此下载所要显示的四大美女的图片)。
首先,先设计一个美女(Girl)类,它是备忘录角色,提供了获取和存储美女信息的功能;然后,设计一个相亲者(You)类,它是发起人角色,它记录当前时刻的内部状态信息(临时妻子的姓名),并提供创建备忘录和恢复备忘录数据的功能;最后,定义一个美女栈(GirlStack)类,它是管理者角色,负责对备忘录进行管理,用于保存相亲者(You)前面选过的美女信息,不过最多只能保存 4 个,提供后悔功能。
客户类设计成窗体程序,它包含美女栈(GirlStack)对象和相亲者(You)对象,它实现了 ActionListener 接口的事件处理方法 actionPerformed(ActionEvent e),并将 4 大美女图像和相亲者(You)选择的美女图像在窗体中显示出来。图 2 所示是其结构图。
图2 相亲游戏的结构图
程序代码如下:
- package net.biancheng.c.memento;
- import javax.swing.*;
- import java.awt.*;
- import java.awt.event.ActionEvent;
- import java.awt.event.ActionListener;
- public class DatingGame {
- public static void main(String[] args) {
- new DatingGameWin();
- }
- }
- //客户窗体类
- class DatingGameWin extends JFrame implements ActionListener {
- private static final long serialVersionUID = 1L;
- JPanel CenterJP, EastJP;
- JRadioButton girl1, girl2, girl3, girl4;
- JButton button1, button2;
- String FileName;
- JLabel g;
- You you;
- GirlStack girls;
- DatingGameWin() {
- super("利用备忘录模式设计相亲游戏");
- you = new You();
- girls = new GirlStack();
- this.setBounds(0, 0, 900, 380);
- this.setResizable(false);
- FileName = "src/memento/Photo/四大美女.jpg";
- g = new JLabel(new ImageIcon(FileName), JLabel.CENTER);
- CenterJP = new JPanel();
- CenterJP.setLayout(new GridLayout(1, 4));
- CenterJP.setBorder(BorderFactory.createTitledBorder("四大美女如下:"));
- CenterJP.add(g);
- this.add("Center", CenterJP);
- EastJP = new JPanel();
- EastJP.setLayout(new GridLayout(1, 1));
- EastJP.setBorder(BorderFactory.createTitledBorder("您选择的爱人是:"));
- this.add("East", EastJP);
- JPanel SouthJP = new JPanel();
- JLabel info = new JLabel("四大美女有“沉鱼落雁之容、闭月羞花之貌”,您选择谁?");
- girl1 = new JRadioButton("西施", true);
- girl2 = new JRadioButton("貂蝉");
- girl3 = new JRadioButton("王昭君");
- girl4 = new JRadioButton("杨玉环");
- button1 = new JButton("确定");
- button2 = new JButton("返回");
- ButtonGroup group = new ButtonGroup();
- group.add(girl1);
- group.add(girl2);
- group.add(girl3);
- group.add(girl4);
- SouthJP.add(info);
- SouthJP.add(girl1);
- SouthJP.add(girl2);
- SouthJP.add(girl3);
- SouthJP.add(girl4);
- SouthJP.add(button1);
- SouthJP.add(button2);
- button1.addActionListener(this);
- button2.addActionListener(this);
- this.add("South", SouthJP);
- showPicture("空白");
- you.setWife("空白");
- girls.push(you.createMemento()); //保存状态
- }
- //显示图片
- void showPicture(String name) {
- EastJP.removeAll(); //清除面板内容
- EastJP.repaint(); //刷新屏幕
- you.setWife(name);
- FileName = "src/memento/Photo/" + name + ".jpg";
- g = new JLabel(new ImageIcon(FileName), JLabel.CENTER);
- EastJP.add(g);
- this.setVisible(true);
- this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
- }
- @Override
- public void actionPerformed(ActionEvent e) {
- boolean ok = false;
- if (e.getSource() == button1) {
- ok = girls.push(you.createMemento()); //保存状态
- if (ok && girl1.isSelected()) {
- showPicture("西施");
- } else if (ok && girl2.isSelected()) {
- showPicture("貂蝉");
- } else if (ok && girl3.isSelected()) {
- showPicture("王昭君");
- } else if (ok && girl4.isSelected()) {
- showPicture("杨玉环");
- }
- } else if (e.getSource() == button2) {
- you.restoreMemento(girls.pop()); //恢复状态
- showPicture(you.getWife());
- }
- }
- }
- //备忘录:美女
- class Girl {
- private String name;
- public Girl(String name) {
- this.name = name;
- }
- public void setName(String name) {
- this.name = name;
- }
- public String getName() {
- return name;
- }
- }
- //发起人:您
- class You {
- private String wifeName; //妻子
- public void setWife(String name) {
- wifeName = name;
- }
- public String getWife() {
- return wifeName;
- }
- public Girl createMemento() {
- return new Girl(wifeName);
- }
- public void restoreMemento(Girl p) {
- setWife(p.getName());
- }
- }
- //管理者:美女栈
- class GirlStack {
- private Girl girl[];
- private int top;
- GirlStack() {
- girl = new Girl[5];
- top = -1;
- }
- public boolean push(Girl p) {
- if (top >= 4) {
- System.out.println("你太花心了,变来变去的!");
- return false;
- } else {
- girl[++top] = p;
- return true;
- }
- }
- public Girl pop() {
- if (top <= 0) {
- System.out.println("美女栈空了!");
- return girl[0];
- } else return girl[top--];
- }
- }
程序运行结果如图 3 所示。
图3 相亲游戏的运行结果(点此查看原图)
前面学习了备忘录模式的定义与特点、结构与实现,现在来看该模式的以下应用场景。
在前面介绍的备忘录模式中,有单状态备份的例子,也有多状态备份的例子。下面介绍备忘录模式如何同原型模式混合使用。在备忘录模式中,通过定义“备忘录”来备份“发起人”的信息,而原型模式的 clone() 方法具有自备份功能,所以,如果让发起人实现 Cloneable 接口就有备份自己的功能,这时可以删除备忘录类,其结构图如图 4 所示。
图4 带原型的备忘录模式的结构图
实现代码如下:
- package net.biancheng.c.memento;
- public class PrototypeMemento {
- public static void main(String[] args) {
- OriginatorPrototype or = new OriginatorPrototype();
- PrototypeCaretaker cr = new PrototypeCaretaker();
- or.setState("S0");
- System.out.println("初始状态:" + or.getState());
- cr.setMemento(or.createMemento()); //保存状态
- or.setState("S1");
- System.out.println("新的状态:" + or.getState());
- or.restoreMemento(cr.getMemento()); //恢复状态
- System.out.println("恢复状态:" + or.getState());
- }
- }
- //发起人原型
- class OriginatorPrototype implements Cloneable {
- private String state;
- public void setState(String state) {
- this.state = state;
- }
- public String getState() {
- return state;
- }
- public OriginatorPrototype createMemento() {
- return this.clone();
- }
- public void restoreMemento(OriginatorPrototype opt) {
- this.setState(opt.getState());
- }
- public OriginatorPrototype clone() {
- try {
- return (OriginatorPrototype) super.clone();
- } catch (CloneNotSupportedException e) {
- e.printStackTrace();
- }
- return null;
- }
- }
- //原型管理者
- class PrototypeCaretaker {
- private OriginatorPrototype opt;
- public void setMemento(OriginatorPrototype opt) {
- this.opt = opt;
- }
- public OriginatorPrototype getMemento() {
- return opt;
- }
- }
程序的运行结果如下:
初始状态:S0 新的状态:S1 恢复状态:S0
拓展
由于 JDK、Spring、Mybatis 中很少有备忘录模式,所以该设计模式不做典型应用源码分析。
Spring Webflow 中 DefaultMessageContext 类实现了 StateManageableMessageContext 接口,查看其源码可以发现其主要逻辑就相当于给 Message 备份。感兴趣的小伙伴可以去阅读学习其源码。
在软件开发中,会遇到有些问题多次重复出现,而且有一定的相似性和规律性。如果将它们归纳成一种简单的语言,那么这些问题实例将是该语言的一些句子,这样就可以用“编译原理”中的解释器模式来实现了。
虽然使用解释器模式的实例不是很多,但对于满足以上特点,且对运行效率要求不是很高的应用实例,如果用解释器模式来实现,其效果是非常好的,本文将介绍其工作原理与使用方法。
解释器(Interpreter)模式的定义:给分析对象定义一个语言,并定义该语言的文法表示,再设计一个解析器来解释语言中的句子。也就是说,用编译语言的方式来分析应用中的实例。这种模式实现了文法表达式处理的接口,该接口解释一个特定的上下文。
这里提到的文法和句子的概念同编译原理中的描述相同,“文法”指语言的语法规则,而“句子”是语言集中的元素。例如,汉语中的句子有很多,“我是中国人”是其中的一个句子,可以用一棵语法树来直观地描述语言中的句子。
解释器模式是一种类行为型模式,其主要优点如下。
解释器模式的主要缺点如下。
解释器模式常用于对简单语言的编译或分析实例中,为了掌握好它的结构与实现,必须先了解编译原理中的“文法、句子、语法树”等相关概念。
1) 文法
文法是用于描述语言的语法结构的形式规则。没有规矩不成方圆,例如,有些人认为完美爱情的准则是“相互吸引、感情专一、任何一方都没有恋爱经历”,虽然最后一条准则较苛刻,但任何事情都要有规则,语言也一样,不管它是机器语言还是自然语言,都有它自己的文法规则。例如,中文中的“句子”的文法如下。
〈句子〉::=〈主语〉〈谓语〉〈宾语〉 〈主语〉::=〈代词〉|〈名词〉 〈谓语〉::=〈动词〉 〈宾语〉::=〈代词〉|〈名词〉 〈代词〉你|我|他 〈名词〉7大学生I筱霞I英语 〈动词〉::=是|学习
注:这里的符号“::=”表示“定义为”的意思,用“〈”和“〉”括住的是非终结符,没有括住的是终结符。
2) 句子
句子是语言的基本单位,是语言集中的一个元素,它由终结符构成,能由“文法”推导出。例如,上述文法可以推出“我是大学生”,所以它是句子。
3) 语法树
语法树是句子结构的一种树型表示,它代表了句子的推导结果,它有利于理解句子语法结构的层次。图 1 所示是“我是大学生”的语法树。
图1 句子“我是大学生”的语法树
有了以上基础知识,现在来介绍解释器模式的结构就简单了。解释器模式的结构与组合模式相似,不过其包含的组成元素比组合模式多,而且组合模式是对象结构型模式,而解释器模式是类行为型模式。
1. 模式的结构
解释器模式包含以下主要角色。
解释器模式的结构图如图 2 所示。
图2 解释器模式的结构图
2. 模式的实现
解释器模式实现的关键是定义文法规则、设计终结符类与非终结符类、画出结构图,必要时构建语法树,其代码结构如下:
- package net.biancheng.c.interpreter;
- //抽象表达式类
- interface AbstractExpression {
- public void interpret(String info); //解释方法
- }
- //终结符表达式类
- class TerminalExpression implements AbstractExpression {
- public void interpret(String info) {
- //对终结符表达式的处理
- }
- }
- //非终结符表达式类
- class NonterminalExpression implements AbstractExpression {
- private AbstractExpression exp1;
- private AbstractExpression exp2;
- public void interpret(String info) {
- //非对终结符表达式的处理
- }
- }
- //环境类
- class Context {
- private AbstractExpression exp;
- public Context() {
- //数据初始化
- }
- public void operation(String info) {
- //调用相关表达式类的解释方法
- }
- }
【例1】用解释器模式设计一个“韶粵通”公交车卡的读卡器程序。
说明:假如“韶粵通”公交车读卡器可以判断乘客的身份,如果是“韶关”或者“广州”的“老人” “妇女”“儿童”就可以免费乘车,其他人员乘车一次扣 2 元。
分析:本实例用“解释器模式”设计比较适合,首先设计其文法规则如下。
<expression> ::= <city>的<person> <city> ::= 韶关|广州 <person> ::= 老人|妇女|儿童
然后,根据文法规则按以下步骤设计公交车卡的读卡器程序的类图。
图3 “韶粵通”公交车读卡器程序的结构图
程序代码如下:
- package net.biancheng.c.interpreter;
- import java.util.*;
- /*文法规则
- <expression> ::= <city>的<person>
- <city> ::= 韶关|广州
- <person> ::= 老人|妇女|儿童
- */
- public class InterpreterPatternDemo {
- public static void main(String[] args) {
- Context bus = new Context();
- bus.freeRide("韶关的老人");
- bus.freeRide("韶关的年轻人");
- bus.freeRide("广州的妇女");
- bus.freeRide("广州的儿童");
- bus.freeRide("山东的儿童");
- }
- }
- //抽象表达式类
- interface Expression {
- public boolean interpret(String info);
- }
- //终结符表达式类
- class TerminalExpression implements Expression {
- private Set<String> set = new HashSet<String>();
- public TerminalExpression(String[] data) {
- for (int i = 0; i < data.length; i++) set.add(data[i]);
- }
- public boolean interpret(String info) {
- if (set.contains(info)) {
- return true;
- }
- return false;
- }
- }
- //非终结符表达式类
- class AndExpression implements Expression {
- private Expression city = null;
- private Expression person = null;
- public AndExpression(Expression city, Expression person) {
- this.city = city;
- this.person = person;
- }
- public boolean interpret(String info) {
- String s[] = info.split("的");
- return city.interpret(s[0]) && person.interpret(s[1]);
- }
- }
- //环境类
- class Context {
- private String[] citys = {"韶关", "广州"};
- private String[] persons = {"老人", "妇女", "儿童"};
- private Expression cityPerson;
- public Context() {
- Expression city = new TerminalExpression(citys);
- Expression person = new TerminalExpression(persons);
- cityPerson = new AndExpression(city, person);
- }
- public void freeRide(String info) {
- boolean ok = cityPerson.interpret(info);
- if (ok) System.out.println("您是" + info + ",您本次乘车免费!");
- else System.out.println(info + ",您不是免费人员,本次乘车扣费2元!");
- }
- }
程序运行结果如下:
您是韶关的老人,您本次乘车免费! 韶关的年轻人,您不是免费人员,本次乘车扣费2元! 您是广州的妇女,您本次乘车免费! 您是广州的儿童,您本次乘车免费! 山东的儿童,您不是免费人员,本次乘车扣费2元!
前面介绍了解释器模式的结构与特点,下面分析它的应用场景。
注意:解释器模式在实际的软件开发中使用比较少,因为它会引起效率、性能以及维护等问题。如果碰到对表达式的解释,在 Java 中可以用 Expression4J 或 Jep 等来设计。
在项目开发中,如果要对数据表达式进行分析与计算,无须再用解释器模式进行设计了,Java 提供了以下强大的数学公式解析器:Expression4J、MESP(Math Expression String Parser) 和 Jep 等,它们可以解释一些复杂的文法,功能强大,使用简单。
现在以 Jep 为例来介绍该工具包的使用方法。Jep 是 Java expression parser 的简称,即 Java 表达式分析器,它是一个用来转换和计算数学表达式的 Java 库。通过这个程序库,用户可以以字符串的形式输入一个任意的公式,然后快速地计算出其结果。而且 Jep 支持用户自定义变量、常量和函数,它包括许多常用的数学函数和常量。
使用前先下载 Jep 压缩包,解压后,将 jep-x.x.x.jar 文件移到选择的目录中,在 Eclipse 的“Java 构建路径”对话框的“库”选项卡中选择“添加外部 JAR(X)...”,将该 Jep 包添加项目中后即可使用其中的类库。
下面以计算存款利息为例来介绍。存款利息的计算公式是:本金x利率x时间=利息,其相关代码如下:
- package net.biancheng.c.interpreter;
- import com.singularsys.jep.*;
- public class JepDemo {
- public static void main(String[] args) throws JepException {
- Jep jep = new Jep();
- //定义要计算的数据表达式
- String 存款利息 = "本金*利率*时间";
- //给相关变量赋值
- jep.addVariable("本金", 10000);
- jep.addVariable("利率", 0.038);
- jep.addVariable("时间", 2);
- jep.parse(存款利息); //解析表达式
- Object accrual = jep.evaluate(); //计算
- System.out.println("存款利息:" + accrual);
- }
- }
程序运行结果如下:
存款利息:760.0
行为型模式(Behavioral Pattern)是对在不同的对象之间划分责任和算法的抽象化,它是 GoF 设计模式中最为庞大的一类模式,包含以下 11 种:模板方法(Template Method)模式、策略(Strategy)模式、命令(Command)模式、职责链(Chain of Responsibility)模式、状态(State)模式、观察者(Observer)模式、中介者(Mediator)模式、迭代器(Iterator)模式、访问者(Visitor)模式、备忘录(Memento)模式、解释器(Interpreter)模式。
本实验的主要目的如下。
1. 行为型模式的工作原理
行为型模式用于描述程序在运行时复杂的流程控制,即描述多个类或对象之间怎样相互协作共同完成单个对象无法单独完成的任务,它涉及算法与对象间职责的分配。
按照其显示方式的不同,行为型模式可分为类行为模式和对象行为模式,其中类行为型模式使用继承关系在几个类之间分配行为,主要通过多态等方式来分配父类与子类的职责;对象行为型模式则使用对象的组合或聚合关联关系来分配行为,主要是通过对象关联等方式来分配两个或多个类的职责。
由于组合关系或聚合关系比继承关系耦合度低,满足“合成复用原则”所以对象行为模式比类行为模式具有更大的灵活性。
如果按目的来分,行为型模式共 11 种,每种模式的工作原理在前面的教程中都有详细的介绍,每种模式的实验大概要花 2 个学时,大家可以根据实验计划来选做若干个实验。下面以观察者模式为例,介绍其实验过程。
2. 观察者模式的工作原理
观察者模式是一种对象行为型模式,用于解决多个对象间存在的一对多的依赖关系。在现实世界中,许多对象并不是独立存在的,其中一个对象的状态发生改变可能会导致一个或者多个其他对象也发生改变,如物价与消费者、股价与股民、天气预报与农民、警察与小偷、事件源与事件处理 者等。这种模式有时又称作发布-订阅模式、模型-视图模式,其结构图如图 1 所示。
图1 观察者模式的结构图
观察者模式包含如下角色。
(1) 用观察者模式设计一个交通信号灯的事件处理程序。
分析:“交通信号灯”是事件源和目标,各种“车”是事件监听器和具体观察者,“信号灯颜色” 是事件类。
(2) 按照以上要求设计类图和编写 Java 源程序。
所设计的实验程序要满足以下两点。
(1) 用 UML 设计“交通信号灯事件处理程序”的结构图。
“交通信号灯事件处理程序”的结构图如图 2 所示。
图2 交通信号灯事件处理程序的结构图
(2) 根据结构图写出“交通信号灯事件处理程序”的源代码。交通信号灯事件处理程序的源代码如下。
- package observer;
- import java.util.*;
- public class SignalLightEvent {
- public static void main(String[] args) {
- SignalLight light = new SignalLight();//交通信号灯(事件源)
- light.addVehicleListener(new Car()); //注册监听器(轿车)
- light.addVehicleListener(new Buses());//注册监听器(公交车)
- light.changeColor("红色");
- System.out.println("------------");
- light.changeColor("绿色");
- }
- }
- //信号灯颜色
- class SignalColor extends EventObject {
- private String color; //"红色"和"绿色"
- public SignalColor(Object source, String color) {
- super(source);
- this.color = color;
- }
- public void setColor(String color) {
- this.color = color;
- }
- public String getColor() {
- return this.color;
- }
- }
- //目标类:事件源,交通信号灯
- class SignalLight {
- private List<Vehicle> listener; //监听器容器
- public SignalLight() {
- listener = new ArrayList<Vehicle>();
- }
- //给事件源绑定监听器
- public void addVehicleListener(Vehicle vehicle) {
- listener.add(vehicle);
- }
- //事件触发器:信号灯改变颜色。
- public void changeColor(String color) {
- System.out.println(color + "信号灯亮...");
- SignalColor event = new SignalColor(this, color);
- notifies(event); //通知注册在该事件源上的所有监听器
- }
- //当事件发生时,通知绑定在该事件源上的所有监听器做出反应(调用事件处理方法)
- protected void notifies(SignalColor e) {
- Vehicle vehicle = null;
- Iterator<Vehicle> iterator = listener.iterator();
- while (iterator.hasNext()) {
- vehicle = iterator.next();
- vehicle.see(e);
- }
- }
- }
- //抽象观察者类:车
- interface Vehicle extends EventListener {
- //事件处理方法,看见
- public void see(SignalColor e);
- }
- //具体观察者类:轿车
- class Car implements Vehicle {
- public void see(SignalColor e) {
- if ("红色".equals(e.getColor())) {
- System.out.println("红灯亮,轿车停!");
- } else {
- System.out.println("绿灯亮,轿车行!");
- }
- }
- }
- //具体观察者类: 公交车
- class Buses implements Vehicle {
- public void see(SignalColor e) {
- if ("红色".equals(e.getColor())) {
- System.out.println("红灯亮,公交车停!");
- } else {
- System.out.println("绿灯亮,公交车行!");
- }
- }
- }
(3) 上机测试程序,写出运行结果。
交通信号灯事件处理程序的运行结果如下:
红色信号灯亮... 红灯亮,轿车停! 红灯亮,公交车停! ------------ 绿色信号灯亮... 绿灯亮,轿车行! 绿灯亮,公交车行!
(4) 按同样的步骤设计其他“观察者模式”的程序实例。
通过《23种设计模式全面解析》教程,我们已经学习完了经典的 23 种设计模式。下面总结一下这 23 种设计模式,以方便小伙伴们日后复习和查阅。
分类 | 设计模式 | 简述 | 一句话归纳 | 目的 | 生活案例 |
---|---|---|---|---|---|
创建型设计模式 (简单来说就是用来创建对象的) | 工厂模式(Factory Pattern) | 不同条件下创建不同实例 | 产品标准化,生产更高效 | 封装创建细节 | 实体工厂 |
单例模式(Singleton Pattern) | 保证一个类仅有一个实例,并且提供一个全局访问点 | 世上只有一个我 | 保证独一无二 | CEO | |
原型模式(Prototype Pattern) | 通过拷贝原型创建新的对象 | 拔一根猴毛,吹出千万个 | 高效创建对象 | 克隆 | |
建造者模式(Builder Pattern) | 用来创建复杂的复合对象 | 高配中配和低配,想选哪配就哪配 | 开放个性配置步骤 | 选配 | |
结构型设计模式 (关注类和对象的组合) | 代理模式(Proxy Pattern) | 为其他对象提供一种代理以控制对这个对象的访问 | 没有资源没时间,得找别人来帮忙 | 增强职责 | 媒婆 |
外观模式(Facade Pattern) | 对外提供一个统一的接口用来访问子系统 | 打开一扇门,通向全世界 | 统一访问入口 | 前台 | |
装饰器模式(Decorator Pattern) | 为对象添加新功能 | 他大舅他二舅都是他舅 | 灵活扩展、同宗同源 | 煎饼 | |
享元模式(Flyweight Pattern) | 使用对象池来减少重复对象的创建 | 优化资源配置,减少重复浪费 | 共享资源池 | 全国社保联网 | |
组合模式(Composite Pattern) | 将整体与局部(树形结构)进行递归组合,让客户端能够以一种的方式对其进行处理 | 人在一起叫团伙,心在一起叫团队 | 统一整体和个体 | 组织架构树 | |
适配器模式(Adapter Pattern) | 将原来不兼容的两个类融合在一起 | 万能充电器 | 兼容转换 | 电源适配 | |
桥接模式(Bridge Pattern) | 将两个能够独立变化的部分分离开来 | 约定优于配置 | 不允许用继承 | 桥 | |
行为型设计模式 (关注对象之间的通信) | 模板模式(Template Pattern) | 定义一套流程模板,根据需要实现模板中的操作 | 流程全部标准化,需要微调请覆盖 | 逻辑复用 | 把大象装进冰箱 |
策略模式(Strategy Pattern) | 封装不同的算法,算法之间能互相替换 | 条条大道通罗马,具体哪条你来定 | 把选择权交给用户 | 选择支付方式 | |
责任链模式(Chain of Responsibility Pattern) | 拦截的类都实现统一接口,每个接收者都包含对下一个接收者的引用。将这些对象连接成一条链,并且沿着这条链传递请求,直到有对象处理它为止。 | 各人自扫门前雪,莫管他们瓦上霜 | 解耦处理逻辑 | 踢皮球 | |
迭代器模式(Iterator Pattern) | 提供一种方法顺序访问一个聚合对象中的各个元素 | 流水线上坐一天,每个包裹扫一遍 | 统一对集合的访问方式 | 逐个检票进站 | |
命令模式(Command Pattern) | 将请求封装成命令,并记录下来,能够撤销与重做 | 运筹帷幄之中,决胜千里之外 | 解耦请求和处理 | 遥控器 | |
状态模式(State Pattern) | 根据不同的状态做出不同的行为 | 状态驱动行为,行为决定状态 | 绑定状态和行为 | 订单状态跟踪 | |
备忘录模式(Memento Pattern) | 保存对象的状态,在需要时进行恢复 | 失足不成千古恨,想重来时就重来 | 备份、后悔机制 | 草稿箱 | |
中介者模式(Mediator Pattern) | 将对象之间的通信关联关系封装到一个中介类中单独处理,从而使其耦合松散 | 联系方式我给你,怎么搞定我不管 | 统一管理网状资源 | 朋友圈 | |
解释器模式(Interpreter Pattern) | 给定一个语言,定义它的语法表示,并定义一个解释器,这个解释器使用该标识来解释语言中的句子 | 我想说”方言“,一切解释权都归我 | 实现特定语法解析 | 摩斯密码 | |
观察者模式(Observer Pattern) | 状态发生改变时通知观察者,一对多的关系 | 到点就通知我 | 解耦观察者与被观察者 | 闹钟 | |
访问者模式(Visitor Pattern) | 稳定数据结构,定义新的操作行为 | 横看成岭侧成峰,远近高低各不同 | 解耦数据结构和数据操作 | KPI考核 | |
委派模式(Delegate Pattern) | 允许对象组合实现与继承相同的代码重用,负责任务的调用和分配 | 这个需求很简单,怎么实现我不管 | 只对结果负责 | 授权委托书 |
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。