[设计模式]组合模式(Compostie)

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;
}


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值