1.意图
将对象组合成树状结构以表示“部分—整体”的层次结构。Composite使得用户对单个对象和组合的使用具有一致性(整体和部分一致对待)。
2.动机
在图形应用系统中可以用简单的组件(图元)创建复杂的组件(图表)。用户可以组合多个组件形成一个较大的组件(Winform开发中就有自定义控件),这些组件又可以组合成更大的组件。一个实现方法就是对组件定义一些类,另外定义一些类作为这些组件的容器类。
然而这带来一个问题:使用这些类的代码必须区别对待组件类和容器类,而实际上大多数情况下用户认为他们是一样的。对这些类的区别使用会使得程序更加复杂。组合模式描述了如何使用递归组合,使得用户不必对这些类进行区别。
3.结构与参与者
4.效果
1)定义了包含基本对象和组合对象的类层次结构
基本对象可以被组合成更复杂的组合对象,而这个组合对象又可以被组合,这样不断地递归下去。客户代码中,任何用到基本对象的地方都可以使用组合对象。
2)简化客户代码
客户可以一致地使用组合结构和单个对象(就好比用自定义控件就不需要每次都从轮子造起了)。
3)使得更容易增加新类型的组件
新定义的Composite或Leaf子类自动地与已有的结构和客户代码一起工作,客户程序不需要因加入新的组件而改变。
4)使设计变得更加一般化(缺)
容易增加新组建也会产生一些问题,那就是很难限制组合中的组件。有时你希望一个组合只能有某些特定组件。使用组合模式是,你不能依赖系统类型施加这些约束,而必须在运行时刻进行检查。
5.实现
6.代码示例
#include <iostream>
#include <string>
#include <vector>
using namespace std;
class Company
{
protected:
string m_strName;
public:
Company(string strName)
{
m_strName = strName;
}
virtual void Add(Company* c)=0;
virtual void Display(int nDepth)=0;
virtual void LineOfDuty()=0;
virtual ~Company(){}
};
class ConcreteCompany: public Company
{
private:
vector<Company*> m_company;
public:
ConcreteCompany(string strName):Company(strName){}
virtual void Add(Company* c)
{
m_company.push_back(c);
}
virtual void Display(int nDepth)
{
string strtemp;
for(int i=0; i < nDepth; i++)
{
strtemp += "-";
}
strtemp +=m_strName;
cout<<strtemp<<endl;
vector<Company*>::iterator p=m_company.begin();
while (p!=m_company.end())
{
(*p)->Display(nDepth+2);
p++;
}
}
virtual void LineOfDuty()
{
vector<Company*>::iterator p=m_company.begin();
while (p!=m_company.end())
{
(*p)->LineOfDuty();
p++;
}
}
};
class HrDepartment : public Company
{
public:
HrDepartment(string strname) : Company(strname){}
virtual void Display(int nDepth)
{
string strtemp;
for(int i = 0; i < nDepth; i++)
{
strtemp += "-";
}
strtemp += m_strName;
cout<<strtemp<<endl;
}
virtual void Add(Company* c)
{
cout<<"error"<<endl;
}
virtual void LineOfDuty()
{
cout<<m_strName<<":招生"<<endl;
}
};
//客户端:
int main()
{
ConcreteCompany *p = new ConcreteCompany("东南大学");
p->Add(new HrDepartment("东南大学招生办"));
ConcreteCompany *p1 = new ConcreteCompany("数学系");
p1->Add(new HrDepartment("数学系招生办"));
ConcreteCompany *p2 = new ConcreteCompany("物理系");
p2->Add(new HrDepartment("物理系招生办"));
p->Add(p1);
p->Add(p2);
cout << "组织结构:"<<endl;
p->Display(1);
cout << "各部门职责:"<<endl;
p->LineOfDuty();
return 0;
}