一.简介
今天来学习一下组合模式。说到组合模式,可能比较陌生,但是有很多地方是非常适合使用组合模式的。比如我们电脑系统内的文件系统。文件包含在文件夹内,而文件夹有可能还被其他文件夹包含,整个文件系统呈一个树形结构。对于文件的操作,我们可以操作整个文件夹,相应的文件夹下所有的子文件,子文件夹都会被递归的操作。而我们并不需要知道文件夹下面是什么东东,只需要对根文件夹操作即可。
下面看一下组合模式的定义以及UML类图:
组合模式(Composite Pattern):组合多个对象形成树形结构以表示具有“整体—部分”关系的层次结构。组合模式对单个对象(即叶子对象)和组合对象(即容器对象)的使用具有一致性,组合模式又可以称为“整体—部分”(Part-Whole)模式,它是一种对象结构型模式。
二.组合模式的例子
好了,要想更加深刻地理解设计模式,还是要通过实际的例子才能获得更深刻的印象。就拿我们最常用的文件系统来说,如果这个文件系统需要我们来实现,那我们需要怎么实现文件系统的结构呢?给文件实现一个最简单的功能。我们知道,文件系统中分为文件夹和具体的文件,而文件夹内可能包含文件,也可能包含文件夹,整体呈现树形结构,下面我们使用代码来实现以下文件系统。
1.没有使用组合模式的情况
// Design Pattern.cpp : Defines the entry point for the console application.
//
#include "stdafx.h"
#include <iostream>
#include <string>
#include <vector>
using namespace std;
//文件类
class TextFile
{
private:
string name;
public:
TextFile(string n)
: name(n)
{
}
//文件操作功能
void Option()
{
cout << name <<" 文件操作" << endl;
}
};
class FileFolder
{
private:
string name;
vector<FileFolder*> fileFolderVec;
vector<TextFile*> textFileVec;
public:
FileFolder(string n)
: name(n)
{
}
void AddFile(TextFile* file)
{
textFileVec.push_back(file);
}
void AddFileFolder(FileFolder* folder)
{
fileFolderVec.push_back(folder);
}
void Option()
{
cout<< name << " 文件夹被操作" << endl;
//先遍历文件容器,将文件夹本身包含的文件进行操作
for (vector<TextFile*>::iterator it = textFileVec.begin(); it != textFileVec.end(); ++it)
{
(*it)->Option();
}
//再遍历文件夹容器,递归调用文件夹下文件夹的操作
for (vector<FileFolder*>::iterator it = fi