//---------------------------15/04/16----------------------------

//Composite 组合模式----对象结构型模式

/*

1:意图

将对象组合成树形结构以表示“部分-整体”的层次结构。Composite使得用户对单个对象和

组合对象的使用具有一致性。

2:动机:

3:适用性:

1>你想表示对象的部分-整体层次结构。

2>你希望用户忽略组合对象与单个对象的不同,用户将统一地使用组合结构中的所有对象。

4:结构:

Client--------->Component:<--------------

Operation()             |

Add(Component)          |

Remove(Component)       |

GetChild(int)           |

|                   |

|                   |

--------------            |

|             |           |

Leaf:       Composite:      |

Operation() children---------

Operation()

{ for all g in children

{ g.Operation()}

}

Add(Comonent)

Remove(Component)

GetChild(int)

5:参与者:

1>Component

1)为组合中的对象声明接口。

2)在适当的情况下,实现所有类共有接口的缺省行为。

3)声明一个接口用于访问和管理Component的子组件。

4)(可选)在递归结构中定义一个接口,用于访问一个父部件,并在合适的情况下实现它。

2>Leaf

1)在组合中表示叶节点对象,叶节点没有子节点。

2)在组合中定义图元对象的行为。

3>Composite

1)定义有子部件的那些部件的行为。

2)存储子部件。

3)在Component接口中实现与子部件有关的操作。

4>Client

通过Component接口操纵组合部件的对象。

6:协作:

用户使用Component类接口与组合结构中的对象进行交互。如果接收者是一个叶节点,则直接处理请求。

如果接受者是Composite,它通常将请求发送给它的子部件,在转发请求之前与(或)之后可能执行一些

辅助操作。

7:效果:

1>定义了包含基本对象和组合对象的类层次结构。

基本对象可以被组合成更复杂的组合对象,而这个组合对象又可以被组合,这样不断递归下去。客户

代码中,任何用到基本对象的地方都可以使用组合对象。

2>简化客户代码。

客户可以一致地使用组合结构和单个对象。通常用户不知道(也不关心)处理的是一个叶节点还是一个

组合对象。这样就简化了客户代码,不必写一大堆选择语句来区分它们。

3>使得更容易增加新类型的组件。

新定义的Composite或Leaf子类自动地与已有的结构和客户代码一起工作,客户程序不需因新增

的Component类而改变。

4>使你的设计变得更加一般化。

容易增加新组件意味着很难限制组合中的组件。如果你希望一个组合只能由一些特定的组件,

在使用Composite模式时却不能依赖类型系统施加这些约束,而必须在运行时刻进行检查。

8:实现:

1>显示的父部件引用。(也就是存储一个父节点的指针)

保持从子部件到父部件的引用能简化组合结构的遍历和管理。父部件引用可以简化结构的上移和

组件的删除,同时也能支持责任链模式。

对于父部件引用必须维护一个不变式,即一个组合的所有子节点以这个组合为父节点,而反之

该组合以这些节点为子节点。可以选择在Composite类的Add和Remove操作中实现这种方法。

2>共享组件:

共享组件可以减少对存储的需求,可以使用Flywight方式来实行共享。

3>最大化Component接口

Composite模式的目的之一是使得用户不知道他们正在使用的具体的Leaf和Composite类。

为了达到这个目的,Component类应该为Leaf和Composite类尽可能多定义一些公共操作。

为了不限制Leaf类,可以对Composite类才需要的
操作实现缺省的操作。

4>声明管理子部件的操作

选择在Component类还是在Composite类中声明管理子部件的操作,这需要在安全性和透明性之间

做出选择:

1)在Component中声明:这样拥有良好的透明性,因为这样可以一致地使用所有组件,但是会付出

安全性的代价,因为客户有可能会在Leaf中做出无意义的操作,比如增加或删除对象。

2)在Composite类中声明:这样做女友良好的安全性,但损失了透明性。接口变得不一致了。

这里我们比较看重透明性:所以要在Component中定义缺省的Add和Remove操作。这时又带来一个问题

客户可能会向一个Leaf中添加或删除组件,所以通常可以使用缺省方式处理Add和Remove的失败。

有一个方式是:产生一个异常,这样客户就能知道失败原因了。

5>Component是否应该实现一个Component列表。

也就是在Component中是否应该存放子节点的指针,这个答案是只有在叶子节点相对很少时才值得

这么做,因为叶子节点没有子节点,存放指针会浪费空间。

6>子部件排序。

如果需要考虑子节点的顺序时,必须仔细地设计对子节点的访问和管理接口,以便管理子节点序列。

可以参考iterator模式。

7>使用高速缓存改善性能。

使用缓存机制可以加速对子部件的访问。但是如果子部件发生了变化,那么这个子部件必须通知他的

父节点高速缓存失效了。所以需要在Compnent中定义一个private接口。

8>应该由谁删除Component

通常最好由Composite负责删除其子节点。Leaf对象不会改变,所以可以被共享而不删除。

9>存贮组件最好用哪一种数据结构

什么效率高就用什么,包括数组,树,hash表,链表。

9:代码示例:                                                                         */

//假设我们要组装一台电脑,这里每个设备只需要知道其价格就行,所以可以设计一个component类

//Compnent类:
给出了设备名 和价格
以及对子部件的操作

class Equipment

{

public:

virtual ~Equipment();

const char* Name() {
return _name;}

virtual Watt Power();

virtual Currency NetPrice();

virtual Currency DiscountPrice();

virtual void Add(Equipment*);

virtual void Remove(Equipment*);

virtual Iterator<Equipment*>* CreateIterator();

protected:

Equipment(const
char*);

private:

const char* _name;

};

//Leaf组件:实现了Compnent类中的组件特有操作

class FloppyDisk :
public Equipment

{

public:

FloppyDisk(const
char*);

virtual ~FloppyDisk();

virtual Watt Power();

virtual Currency NetPrice();

virtual Currency DiscountPrice();

};

//Composite类:实现了对子部件的操作

class CompositeEquipment :
public Equipment

{

public:

virtual ~CompositeEquipment();

virtual Watt Power();

virtual Currency NetPrice();

virtual Currency DiscountPrice();

virtual void Add(Equipment*);

virtual void Remove(Equipment*);

virtual Iterator<Equipment*>* CreateIterator();

protected:

CompositeEquipment(const
char*);

private:

List<Equipment*> _equipment;

};

//具体的实现,统计每个子部件的价格

Currency CompositeEquipment::NetPrice()

{

Iterator<Equipment*>* i = CreateIterator();

Currency total =
;

for(i->First(); !i->IsDone(); i->Next())

{

total += i->CurrentItem()->NetPrice();

}

delete i;

return total;

}

//具体的Composite类,主板

class Chassis :
public CompositeEquipment

{

public:

Chassis(const
char*);

virtual ~Chassis();

virtual Watt Power();

virtual Currency NetPrice();

virtual Currency DiscountPrice();

};

//机箱

Cabinet* cabinet =
new Cabinet("PC Cabinet");

Chassis* chassis =
new Chassis("PC Chassis");

cabinet->Add(chassis);

//适配卡bus

Bus* bus =
new Bus("MCA Bus");

//加了张网卡

bus->Add(new Card("16Mbs Token Ring"));

chassis->Add(bus);

//加了张软盘

chassis->Add(new FloppyDisk("3.5in Floppy"));

cout<<"Thee net Price is"<<chassis->NetPrice()<<endl;

设计模式 笔记 组合模式 Composite的更多相关文章

  1. 乐在其中设计模式(C#) - 组合模式(Composite Pattern)

    原文:乐在其中设计模式(C#) - 组合模式(Composite Pattern) [索引页][源码下载] 乐在其中设计模式(C#) - 组合模式(Composite Pattern) 作者:weba ...

  2. 二十四种设计模式:组合模式(Composite Pattern)

    组合模式(Composite Pattern) 介绍将对象组合成树形结构以表示"部分-整体"的层次结构.它使得客户对单个对象和复合对象的使用具有一致性.示例有一个Message实体 ...

  3. 【设计模式】组合模式 Composite Pattern

    树形结构是软件行业很常见的一种结构,几乎随处可见,  比如: HTML 页面中的DOM,产品的分类,通常一些应用或网站的菜单,Windows Form 中的控件继承关系,Android中的View继承 ...

  4. [设计模式] 8 组合模式 Composite

    DP书上给出的定义:将对象组合成树形结构以表示“部分-整体”的层次结构.组合使得用户对单个对象和组合对象的使用具有一致性.注意两个字“树形”.这种树形结构在现实生活中随处可见,比如一个集团公司,它有一 ...

  5. python 设计模式之组合模式Composite Pattern

    #引入一 文件夹对我们来说很熟悉,文件夹里面可以包含文件夹,也可以包含文件. 那么文件夹是个容器,文件夹里面的文件夹也是个容器,文件夹里面的文件是对象. 这是一个树形结构 咱们生活工作中常用的一种结构 ...

  6. 设计模式-12组合模式(Composite Pattern)

    1.模式动机 很多时候会存在"部分-整体"的关系,例如:大学中的部门与学院.总公司中的部门与分公司.学习用品中的书与书包.在软件开发中也是这样,例如,文件系统中的文件与文件夹.窗体 ...

  7. 【设计模式】—— 组合模式Composite

    前言:[模式总览]——————————by xingoo 模式意图 使对象组合成树形的结构.使用户对单个对象和组合对象的使用具有一致性. 应用场景 1 表示对象的 部分-整体 层次结构 2 忽略组合对 ...

  8. 结构型设计模式之组合模式(Composite)

    结构 意图 将对象组合成树形结构以表示“部分-整体”的层次结构.C o m p o s i t e 使得用户对单个对象和组合对象的使用具有一致性. 适用性 你想表示对象的部分-整体层次结构. 你希望用 ...

  9. 设计模式之组合模式(Composite)

    组合模式原理:组合模式的作用是讲继承同一父类的不同子类对象组合起来,形成一个树形的结构,例如公司的部门组织 代码如下 #include <iostream> #include <st ...

随机推荐

  1. .NET笔试题集(四)

    转载于:http://www.cnblogs.com/ForEvErNoME/archive/2012/09/10/2678727.html 1.请你简单的说明数据库建立索引的优缺点 使用索引可以加快 ...

  2. SQL Server中数据库文件的存放方式,文件和文件组 (转载)

    简介 在SQL SERVER中,数据库在硬盘上的存储方式和普通文件在Windows中的存储方式没有什么不同,仅仅是几个文件而已.SQL SERVER通过管理逻辑上的文件组的方式来管理文件.理解文件和文 ...

  3. 使用动态SQL创建数据库

    /*其实我也搞不懂为什么要用SQL来创建,明明SQL Server有图形化创建数据库多省事啊!*/USE master; ​DECLARE @sqlstr nvarchar(max)/*定义一个变量* ...

  4. UNIX高级环境编程(10)进程控制(Process Control)- 竞态条件,exec函数,解释器文件和system函数

    本篇主要介绍一下几个内容: 竞态条件(race condition) exec系函数 解释器文件    1 竞态条件(Race Condition) 竞态条件:当多个进程共同操作一个数据,并且结果依赖 ...

  5. Excel表设置加密

    如果你是从事会计或者人事等数据敏感岗位时,有时手上的电子表格不想被其他人浏览或者增删时,可以给表设置一个密码,这样就安全了. 具体操作如下:(以2007版本为例) 01.把需要加密的文件另存为 02. ...

  6. OpenGL超级宝典笔记——贝塞尔曲线和曲面(转)

    http://my.oschina.net/sweetdark/blog/183721 参数方程表现形式 在中学的时候,我们都学习过直线的参数方程:y = kx + b;其中k表示斜率,b表示截距(即 ...

  7. 不使用 vue-cli 与 vue 模版,使用 Vue2.x + webpack4.x 从零开始一步步搭建项目框架

    说明 这是我根据慕课网上的一个课程 Vue+Webpack打造todo应用 过程一步步搭下来的框架,去掉了业务相关的逻辑. 项目最终的效果包括了引入vue框架:使用CSS预处理器:使用babel:引用 ...

  8. C#中单问号(?)和双问号(??)的用法整理

    1.单问号(?) 1.1 表示Nullable类型 C#2.0里面实现了Nullable数据类型 //A.比如下面一句,直接定义int为null是错误的,错误提示为无法将null转化成int,因为后者 ...

  9. Programming Assignment 3: Pattern Recognition

    编程作业三 作业链接:Pattern Recognition & Checklist 我的代码:BruteCollinearPoints.java & FastCollinearPoi ...

  10. Dijkstra(最短路求解)

    Dijkstra(最短路求解) 模板: #include<iostream> #include<cstdio> #include<cstring> #include ...