七个结构型模式之3(适配器 /桥接模式/组合模式)
适配器模式
Sunny软件公司在很久以前曾开发了一个算法库,里面包含了一些常用的算法,例如排序算法和查找算法,在进行各类软件开发时经常需要重用该算法库中的算法。在为某学校开发教务管理系统时,开发人员发现需要对学生成绩进行排序和查找,该系统的设计人员已经开发了一个成绩操作接口ScoreOperation,在该接口中声明了排序方法sort(int[]) 和查找方法search(int[], int),为了提高排序和查找的效率,开发人员决定重用算法库中的快速排序算法类QuickSort和二分查找算法类BinarySearch,其中QuickSort的quickSort(int[])方法实现了快速排序,BinarySearch 的binarySearch (int[], int)方法实现了二分查找。
由于某些原因,现在Sunny公司开发人员已经找不到该算法库的源代码,无法直接通过复制和粘贴操作来重用其中的代码;部分开发人员已经针对ScoreOperation接口编程,如果再要求对该接口进行修改或要求大家直接使用QuickSort类和BinarySearch类将导致大量代码需要修改。
Sunny软件公司开发人员面对这个没有源码的算法库,遇到一个幸福而又烦恼的问题:如何在既不修改现有接口又不需要任何算法库代码的基础上能够实现算法库的重用?
//抽象成绩操作类:目标接口
interface ScoreOperation {
public int[] sort(int array[]); //成绩排序
public int search(int array[],int key); //成绩查找
}
//快速排序类:适配者
class QuickSort {
public int[] quickSort(int array[]) {
sort(array,0,array.length-1);
return array;
}
public void sort(int array[],int p, int r) {
int q=0;
if(p
q=partition(array,p,r);
sort(array,p,q-1);
sort(array,q+1,r);
}
}
public int partition(int[] a, int p, int r) {
int x=a[r];
int j=p-1;
for (int i=p;i<=r-1;i++) {
if (a[i]<=x) {
j++;
swap(a,j,i);
}
}
swap(a,j+1,r);
return j+1;
}
public void swap(int[] a, int i, int j) {
int t = a[i];
a[i] = a[j];
a[j] = t;
}
}
//二分查找类:适配者
class BinarySearch {
public int binarySearch(int array[],int key) {
int low = 0;
int high = array.length -1;
while(low <= high) {
int mid = (low + high) / 2;
int midVal = array[mid];
if(midVal < key) {
low = mid +1;
}
else if (midVal > key) {
high = mid -1;
}
else {
return 1; //找到元素返回1
}
}
return -1; //未找到元素返回-1
}
}
//操作适配器:适配器
class OperationAdapter implements ScoreOperation {
private QuickSort sortObj; //定义适配者QuickSort对象
private BinarySearch searchObj; //定义适配者BinarySearch对象
public OperationAdapter() {
sortObj = new QuickSort();
searchObj = new BinarySearch();
}
public int[] sort(int array[]) {
return sortObj.quickSort(array); //调用适配者类QuickSort的排序方法
}
public int search(int array[],int key) {
return searchObj.binarySearch(array,key); //调用适配者类BinarySearch的查找方法
}
}
缺省适配器模式(Default Adapter Pattern):当不需要实现一个接口所提供的所有方法时,可先设计一个抽象类实现该接口,并为接口中每个方法提供一个默认实现(空方法),那么该抽象类的子类可以选择性地覆盖父类的某些方法来实现需求,它适用于不想使用一个接口中的所有方法的情况,又称为单接口适配器模式。
桥接模式
Sunny软件公司欲开发一个跨平台图像浏览系统,要求该系统能够显示BMP、JPG、GIF、PNG等多种格式的文件,并且能够在Windows、Linux、Unix等多个操作系统上运行。系统首先将各种格式的文件解析为像素矩阵(Matrix),然后将像素矩阵显示在屏幕上,在不同的操作系统中可以调用不同的绘制函数来绘制像素矩阵。系统需具有较好的扩展性以支持新的文件格式和操作系统。
答案:
为了减少所需生成的子类数目,实现将操作系统和图像文件格式两个维度分离,使它们可以独立改变,Sunny公司开发人员使用桥接模式来重构跨平台图像浏览系统的设计,其基本结构如图10-5所示:
在图10-5中,Image充当抽象类,其子类JPGImage、PNGImage、BMPImage和GIFImage充当扩充抽象类;ImageImp充当实现类接口,其子类WindowsImp、LinuxImp和UnixImp充当具体实现类。完整代码如下所示:
[java] view plain copy
- //像素矩阵类:辅助类,各种格式的文件最终都被转化为像素矩阵,不同的操作系统提供不同的方式显示像素矩阵
- class Matrix {
- //此处代码省略
- }
-
- //抽象图像类:抽象类
- abstract class Image {
- protected ImageImp imp;
-
- public void setImageImp(ImageImp imp) {
- this.imp = imp;
- }
-
- public abstract void parseFile(String fileName);
- }
-
- //抽象操作系统实现类:实现类接口
- interface ImageImp {
- public void doPaint(Matrix m); //显示像素矩阵m
- }
-
- //Windows操作系统实现类:具体实现类
- class WindowsImp implements ImageImp {
- public void doPaint(Matrix m) {
- //调用Windows系统的绘制函数绘制像素矩阵
- System.out.print("在Windows操作系统中显示图像:");
- }
- }
-
- //Linux操作系统实现类:具体实现类
- class LinuxImp implements ImageImp {
- public void doPaint(Matrix m) {
- //调用Linux系统的绘制函数绘制像素矩阵
- System.out.print("在Linux操作系统中显示图像:");
- }
- }
-
- //Unix操作系统实现类:具体实现类
- class UnixImp implements ImageImp {
- public void doPaint(Matrix m) {
- //调用Unix系统的绘制函数绘制像素矩阵
- System.out.print("在Unix操作系统中显示图像:");
- }
- }
-
- //JPG格式图像:扩充抽象类
- class JPGImage extends Image {
- public void parseFile(String fileName) {
- //模拟解析JPG文件并获得一个像素矩阵对象m;
- Matrix m = new Matrix();
- imp.doPaint(m);
- System.out.println(fileName + ",格式为JPG。");
- }
- }
-
- //PNG格式图像:扩充抽象类
- class PNGImage extends Image {
- public void parseFile(String fileName) {
- //模拟解析PNG文件并获得一个像素矩阵对象m;
- Matrix m = new Matrix();
- imp.doPaint(m);
- System.out.println(fileName + ",格式为PNG。");
- }
- }
-
- //BMP格式图像:扩充抽象类
- class BMPImage extends Image {
- public void parseFile(String fileName) {
- //模拟解析BMP文件并获得一个像素矩阵对象m;
- Matrix m = new Matrix();
- imp.doPaint(m);
- System.out.println(fileName + ",格式为BMP。");
- }
- }
-
- //GIF格式图像:扩充抽象类
- class GIFImage extends Image {
- public void parseFile(String fileName) {
- //模拟解析GIF文件并获得一个像素矩阵对象m;
- Matrix m = new Matrix();
- imp.doPaint(m);
- System.out.println(fileName + ",格式为GIF。");
- }
- }
为了让系统具有更好的灵活性和可扩展性,我们引入了配置文件,将具体扩充抽象类和具体实现类类名都存储在配置文件中,再通过反射生成对象,将生成的具体实现类对象注入到扩充抽象类对象中,其中,配置文件config.xml的代码如下所示:
[html] view plain copy
-
-
-
- JPGImage
-
- WindowsImp
-
用于读取配置文件config.xml并反射生成对象的XMLUtil类的代码如下所示:
[java] view plain copy
- import javax.xml.parsers.*;
- import org.w3c.dom.*;
- import org.xml.sax.SAXException;
- import java.io.*;
- public class XMLUtil {
- //该方法用于从XML配置文件中提取具体类类名,并返回一个实例对象
- public static Object getBean(String args) {
- try {
- //创建文档对象
- DocumentBuilderFactory dFactory = DocumentBuilderFactory.newInstance();
- DocumentBuilder builder = dFactory.newDocumentBuilder();
- Document doc;
- doc = builder.parse(new File("config.xml"));
- NodeList nl=null;
- Node classNode=null;
- String cName=null;
- nl = doc.getElementsByTagName("className");
-
- if(args.equals("image")) {
- //获取第一个包含类名的节点,即扩充抽象类
- classNode=nl.item(0).getFirstChild();
-
- }
- else if(args.equals("os")) {
- //获取第二个包含类名的节点,即具体实现类
- classNode=nl.item(1).getFirstChild();
- }
-
- cName=classNode.getNodeValue();
- //通过类名生成实例对象并将其返回
- Class c=Class.forName(cName);
- Object obj=c.newInstance();
- return obj;
- }
- catch(Exception e) {
- e.printStackTrace();
- return null;
- }
- }
- }
编写如下客户端测试代码:
[java] view plain copy
- class Client {
- public static void main(String args[]) {
- Image image;
- ImageImp imp;
- image = (Image)XMLUtil.getBean("image");
- imp = (ImageImp)XMLUtil.getBean("os");
- image.setImageImp(imp);
- image.parseFile("小龙女");
- }
- }
组合模式:
Sunny软件公司欲开发一个杀毒(AntiVirus)软件,该软件既可以对某个文件夹(Folder)杀毒,也可以对某个指定的文件(File)进行杀毒。该杀毒软件还可以根据各类文件的特点,为不同类型的文件提供不同的杀毒方式,例如图像文件(ImageFile)和文本文件(TextFile)的杀毒方式就有所差异。现需要提供该杀毒软件的整体框架设计方案
答案:
[java] view plain copy
- //为了突出核心框架代码,我们对杀毒过程的实现进行了大量简化
- import java.util.*;
-
- //图像文件类
- class ImageFile {
- private String name;
-
- public ImageFile(String name) {
- this.name = name;
- }
-
- public void killVirus() {
- //简化代码,模拟杀毒
- System.out.println("----对图像文件'" + name + "'进行杀毒");
- }
- }
-
- //文本文件类
- class TextFile {
- private String name;
-
- public TextFile(String name) {
- this.name = name;
- }
-
- public void killVirus() {
- //简化代码,模拟杀毒
- System.out.println("----对文本文件'" + name + "'进行杀毒");
- }
- }
-
- //文件夹类
- class Folder {
- private String name;
- //定义集合folderList,用于存储Folder类型的成员
- private ArrayList folderList = new ArrayList();
- //定义集合imageList,用于存储ImageFile类型的成员
- private ArrayList imageList = new ArrayList();
- //定义集合textList,用于存储TextFile类型的成员
- private ArrayList textList = new ArrayList();
-
- public Folder(String name) {
- this.name = name;
- }
-
- //增加新的Folder类型的成员
- public void addFolder(Folder f) {
- folderList.add(f);
- }
-
- //增加新的ImageFile类型的成员
- public void addImageFile(ImageFile image) {
- imageList.add(image);
- }
-
- //增加新的TextFile类型的成员
- public void addTextFile(TextFile text) {
- textList.add(text);
- }
-
- //需提供三个不同的方法removeFolder()、removeImageFile()和removeTextFile()来删除成员,代码省略
-
- //需提供三个不同的方法getChildFolder(int i)、getChildImageFile(int i)和getChildTextFile(int i)来获取成员,代码省略
-
- public void killVirus() {
- System.out.println("****对文件夹'" + name + "'进行杀毒"); //模拟杀毒
-
- //如果是Folder类型的成员,递归调用Folder的killVirus()方法
- for(Object obj : folderList) {
- ((Folder)obj).killVirus();
- }
-
- //如果是ImageFile类型的成员,调用ImageFile的killVirus()方法
- for(Object obj : imageList) {
- ((ImageFile)obj).killVirus();
- }
-
- //如果是TextFile类型的成员,调用TextFile的killVirus()方法
- for(Object obj : textList) {
- ((TextFile)obj).killVirus();
- }
- }
- }
编写如下客户端测试代码进行测试:
[java] view plain copy
- class Client {
- public static void main(String args[]) {
- Folder folder1,folder2,folder3;
- folder1 = new Folder("Sunny的资料");
- folder2 = new Folder("图像文件");
- folder3 = new Folder("文本文件");
-
- ImageFile image1,image2;
- image1 = new ImageFile("小龙女.jpg");
- image2 = new ImageFile("张无忌.gif");
-
- TextFile text1,text2;
- text1 = new TextFile("九阴真经.txt");
- text2 = new TextFile("葵花宝典.doc");
-
- folder2.addImageFile(image1);
- folder2.addImageFile(image2);
- folder3.addTextFile(text1);
- folder3.addTextFile(text2);
- folder1.addFolder(folder2);
- folder1.addFolder(folder3);
-
- folder1.killVirus();
- }
- }
现在这个设计 不符合开闭 原则,如果新增加了 video 类型的文件,就需要改 文件夹的代码,怎么办?
在图11-5中, AbstractFile充当抽象构件类,Folder充当容器构件类,ImageFile、TextFile和VideoFile充当叶子构件类。完整代码如下所示:
[java] view plain copy
- import java.util.*;
-
- //抽象文件类:抽象构件
- abstract class AbstractFile {
- public abstract void add(AbstractFile file);
- public abstract void remove(AbstractFile file);
- public abstract AbstractFile getChild(int i);
- public abstract void killVirus();
- }
-
- //图像文件类:叶子构件
- class ImageFile extends AbstractFile {
- private String name;
-
- public ImageFile(String name) {
- this.name = name;
- }
-
- public void add(AbstractFile file) {
- System.out.println("对不起,不支持该方法!");
- }
-
- public void remove(AbstractFile file) {
- System.out.println("对不起,不支持该方法!");
- }
-
- public AbstractFile getChild(int i) {
- System.out.println("对不起,不支持该方法!");
- return null;
- }
-
- public void killVirus() {
- //模拟杀毒
- System.out.println("----对图像文件'" + name + "'进行杀毒");
- }
- }
-
- //文本文件类:叶子构件
- class TextFile extends AbstractFile {
- private String name;
-
- public TextFile(String name) {
- this.name = name;
- }
-
- public void add(AbstractFile file) {
- System.out.println("对不起,不支持该方法!");
- }
-
- public void remove(AbstractFile file) {
- System.out.println("对不起,不支持该方法!");
- }
-
- public AbstractFile getChild(int i) {
- System.out.println("对不起,不支持该方法!");
- return null;
- }
-
- public void killVirus() {
- //模拟杀毒
- System.out.println("----对文本文件'" + name + "'进行杀毒");
- }
- }
-
- //视频文件类:叶子构件
- class VideoFile extends AbstractFile {
- private String name;
-
- public VideoFile(String name) {
- this.name = name;
- }
-
- public void add(AbstractFile file) {
- System.out.println("对不起,不支持该方法!");
- }
-
- public void remove(AbstractFile file) {
- System.out.println("对不起,不支持该方法!");
- }
-
- public AbstractFile getChild(int i) {
- System.out.println("对不起,不支持该方法!");
- return null;
- }
-
- public void killVirus() {
- //模拟杀毒
- System.out.println("----对视频文件'" + name + "'进行杀毒");
- }
- }
-
- //文件夹类:容器构件
- class Folder extends AbstractFile {
- //定义集合fileList,用于存储AbstractFile类型的成员
- private ArrayList fileList=new ArrayList();
- private String name;
-
- public Folder(String name) {
- this.name = name;
- }
-
- public void add(AbstractFile file) {
- fileList.add(file);
- }
-
- public void remove(AbstractFile file) {
- fileList.remove(file);
- }
-
- public AbstractFile getChild(int i) {
- return (AbstractFile)fileList.get(i);
- }
-
- public void killVirus() {
- System.out.println("****对文件夹'" + name + "'进行杀毒"); //模拟杀毒
-
- //递归调用成员构件的killVirus()方法
- for(Object obj : fileList) {
- ((AbstractFile)obj).killVirus();
- }
- }
- }
编写如下客户端测试代码:
[java] view plain copy
- class Client {
- public static void main(String args[]) {
- //针对抽象构件编程
- AbstractFile file1,file2,file3,file4,file5,folder1,folder2,folder3,folder4;
-
- folder1 = new Folder("Sunny的资料");
- folder2 = new Folder("图像文件");
- folder3 = new Folder("文本文件");
- folder4 = new Folder("视频文件");
-
- file1 = new ImageFile("小龙女.jpg");
- file2 = new ImageFile("张无忌.gif");
- file3 = new TextFile("九阴真经.txt");
- file4 = new TextFile("葵花宝典.doc");
- file5 = new VideoFile("笑傲江湖.rmvb");
-
- folder2.add(file1);
- folder2.add(file2);
- folder3.add(file3);
- folder3.add(file4);
- folder4.add(file5);
- folder1.add(folder2);
- folder1.add(folder3);
- folder1.add(folder4);
-
- //从“Sunny的资料”节点开始进行杀毒操作
- folder1.killVirus();
- }
- }
透明组合模式与安全组合模式
通过引入组合模式,Sunny公司设计的杀毒软件具有良好的可扩展性,在增加新的文件类型时,无须修改现有类库代码,只需增加一个新的文件类作为AbstractFile类的子类即可,但是由于在AbstractFile中声明了大量用于管理和访问成员构件的方法,例如add()、remove()等方法,我们不得不在新增的文件类中实现这些方法,提供对应的错误提示和异常处理。为了简化代码,我们有以下两个解决方案:
解决方案一:将叶子构件的add()、remove()等方法的实现代码移至AbstractFile类中,由AbstractFile提供统一的默认实现,代码如下所示:
[java] view plain copy
- //提供默认实现的抽象构件类
- abstract class AbstractFile {
- public void add(AbstractFile file) {
- System.out.println("对不起,不支持该方法!");
- }
-
- public void remove(AbstractFile file) {
- System.out.println("对不起,不支持该方法!");
- }
-
- public AbstractFile getChild(int i) {
- System.out.println("对不起,不支持该方法!");
- return null;
- }
-
- public abstract void killVirus();
- }
如果客户端代码针对抽象类AbstractFile编程,在调用文件对象的这些方法时将出现错误提示。如果不希望出现任何错误提示,我们可以在客户端定义文件对象时不使用抽象层,而直接使用具体叶子构件本身,客户端代码片段如下所示:
[java] view plain copy
- class Client {
- public static void main(String args[]) {
- //不能透明处理叶子构件
- ImageFile file1,file2;
- TextFile file3,file4;
- VideoFile file5;
- AbstractFile folder1,folder2,folder3,folder4;
- //其他代码省略
- }
- }
这样就产生了一种不透明的使用方式,即在客户端不能全部针对抽象构件类编程,需要使用具体叶子构件类型来定义叶子对象。
解决方案二:除此之外,还有一种解决方法是在抽象构件AbstractFile中不声明任何用于访问和管理成员构件的方法,代码如下所示:
[java] view plain copy
- abstract class AbstractFile {
- public abstract void killVirus();
- }
此时,由于在AbstractFile中没有声明add()、remove()等访问和管理成员的方法,其叶子构件子类无须提供实现;而且无论客户端如何定义叶子构件对象都无法调用到这些方法,不需要做任何错误和异常处理,容器构件再根据需要增加访问和管理成员的方法,但这时候也存在一个问题:客户端不得不使用容器类本身来声明容器构件对象,否则无法访问其中新增的add()、remove()等方法,如果客户端一致性地对待叶子和容器,将会导致容器构件的新增对客户端不可见,客户端代码对于容器构件无法再使用抽象构件来定义,客户端代码片段如下所示:
[java] view plain copy
- class Client {
- public static void main(String args[]) {
-
- AbstractFile file1,file2,file3,file4,file5;
- Folder folder1,folder2,folder3,folder4; //不能透明处理容器构件
- //其他代码省略
- }
- }
在使用组合模式时,根据抽象构件类的定义形式,我们可将组合模式分为透明组合模式和安全组合模式两种形式:
(1) 透明组合模式
透明组合模式中,抽象构件Component中声明了所有用于管理成员对象的方法,包括add()、remove()以及getChild()等方法,这样做的好处是确保所有的构件类都有相同的接口。在客户端看来,叶子对象与容器对象所提供的方法是一致的,客户端可以相同地对待所有的对象。透明组合模式也是组合模式的标准形式,虽然上面的解决方案一在客户端可以有不透明的实现方法,但是由于在抽象构件中包含add()、remove()等方法,因此它还是透明组合模式,透明组合模式的完整结构如图11-6所示: