组合模式(Composite)属于结构性模式,它描述了对象间的组合关系。
对象间常常通过树结构来组织(包含)起来,以实现整体-部分的层次结构。整体上可以看做是一个组合对象。
抛却各种复杂的术语,组合模式的特点是:
对象通过实现(继承)统一的接口(抽象类),调用者对单一对象和组合对象的操作具有一致性。
组合模式很常见,Java的容器类(比如ArrayList)都实现了组合模式。
我们平常操作最多的文件(夹)就是组合模式的最好例子。文件系统是树形层次结构,并且文件和文件夹提供给用户的一些操作是相同的,比如复制。于是,可以通过一个统一的接口将文件和文件夹统一起来,对用户提供一致的操作,屏蔽不同的复制实现过程。我们在复制文件夹的时候,操作系统实现了对文件夹内的所有文件和文件夹的复制,即实现了组合对象的整体复制,而不是一个空的文件夹;这和我们复制单个文件的体验是一致的。这便是组合模式的妙处。
完整的模拟代码如下:
- package com.csufox.Composite;
- import java.util.ArrayList;
- interface Node{
- public void copy(); //定义统一的接口:复制
- }
- class Folder implements Node{
- private String folderName;
- private ArrayList<Node> nodeList =new ArrayList<Node>(); //用于存储文件夹下的文件夹或文件的信息
- public Folder(String folderName){
- this.folderName = folderName;
- }
- public void add(Node node){ //增加文件或文件夹
- nodeList.add(node);
- }
- public void copy(){ //文件夹复制操作实现递归
- System.out.println("复制文件夹:" + folderName);
- for(int i=0;i<nodeList.size();i++){
- Node node = (Node)nodeList.get(i);
- node.copy();
- }
- }
- }
- class File implements Node{
- private String fileName;
- public File(String fileName){
- this.fileName = fileName;
- }
- public void copy(){
- System.out.println("复制文件:" + fileName);
- }
- }
- public class Composite{
- public static void main(String[] args){
- Folder document = new Folder("我的资料"); //我的资料文件夹
- File book = new File("Java编程思想.pdf"); //文档文件
- Folder music = new Folder("我的音乐"); //我的音乐文件夹
- File music1 = new File("你是我的眼.mp3"); //音乐文件1
- File music2 = new File("Without You.mp3"); //音乐文件2
- //确定树形结构关系
- document.add(book);
- document.add(music);
- music.add(music1);
- music.add(music2);
- document.copy(); //复制“我的资料”文件夹,递归地复制了其下所有文件夹和文件。
- }
- }
运行结果如下:
- 复制文件夹:我的资料
- 复制文件:Java编程思想.pdf
- 复制文件夹:我的音乐
- 复制文件:你是我的眼.mp3
- 复制文件:Without You.mp3
由以上的代码和运行结果可知:
通过实现组合模式,用户对文件夹的操作与对普通文件的操作并无差异。用户完全不用关心这是文件夹还是文件,也不用关心文件夹内部的具体结构,就可以完成相关操作。
同样的道理,我们可以表达如下:
通过实现组合模式,调用者对组合对象的操作与对单一对象的操作具有一致性。调用者不用关心这是组合对象还是文件,也不用关心组合对象内部的具体结构,就可以调用相关方法,实现功能。
仔细分析copy()方法的代码,我们会发现,如果从面向过程的角度思考,组合模式通过递归原理实现了树结构(组合对象)的深度优先遍历。