23种设计模式汇总结构型模式-组合模式_第1页
23种设计模式汇总结构型模式-组合模式_第2页
23种设计模式汇总结构型模式-组合模式_第3页
23种设计模式汇总结构型模式-组合模式_第4页
23种设计模式汇总结构型模式-组合模式_第5页
已阅读5页,还剩7页未读 继续免费阅读

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

1、组合模式概述组合模式有时候又叫做部分-整体模式,它使我们树型结构的问题中,模糊了简单元素和复杂元素的概念,客户程序可以向处理简单元素一样来处理复杂元素,从而使得客户程序与复杂元素的内部结构解耦。意图将对象组合成树形结构以表示部分-整体”的层次结构oComposite模式使得用户对单个对象和组合对象的使用具有一致性。GOF设计模式结构图fcwaachchildinchildrnchild.图1Composite模式结构图生活中的例子组合模式将对象组合成树形结构以表示部分-整体的层次结构。让用户一致地使用单个对象和组合对象。虽然例子抽象一些,但是算术表达式确实是组合的例子。算术表达式包括操作数、操

2、作符和另一个操作数。操作数可以是数字,也可以是另一个表达式。这样,2+3和(2+3)+(4*6)都是合法的表达式。葺术赵式ooooO图2使用算术表达式例子的Composite模式对象图组合模式解说这里我们用绘图这个例子来说明Composite模式,通过一些基本图像元素(直线、圆等)以及一些复合图像元素(由基本图像元素组合而成)构建复杂的图形树。在设计中我们对每一个对象都配备一个Draw()方法,在调用时,会显示相关的图形。可以看到,这里复合图像元素它在充当对象的同时,又是那些基本图像元素的一个容器。先看一下基本的类结构图:图3图中橙色的区域表示的是复合图像元素。示意性代码:publicabst

3、ractclassGraphicsprotectedstring_name;publicGraphics(stringname)this._name=name;publicabstractvoidDraw();publicclassPicture:GraphicspublicPicture(stringname):base(name)publicoverridevoidDraw()/publicArrayListGetChilds()/返回所有的子对象而其他作为树枝构件,实现代码如下:publicclassLine:GraphicspublicLine(stringname):base(nam

4、e)publicoverridevoidDraw()Console.WriteLine(Drawa+_name.ToString();publicclassCircle:GraphicspublicCircle(stringname):base(name)publicoverridevoidDraw()Console.WriteLine(Drawa+_name.ToString();publicclassRectangle:GraphicspublicRectangle(stringname):base(name)publicoverridevoidDraw()Console.WriteLin

5、e(Drawa+_name.ToString();现在我们要对该图像元素进行处理:在客户端程序中,需要判断返回对象的具体类型到底是基本图像元素,还是复合图像元素。如果是复合图像元素,我们将要用递归去处理,然而这种处理的结果却增加了客户端程序与复杂图像元素内部结构之间的依赖,那么我们如何去解耦这种关系呢?我们希望的是客户程序可以像处理基本图像元素一样来处理复合图像元素,这就要引入Composite模式了,需要把对于子对象的管理工作交给复合图像元素,为了进行子对象的管理,它必须提供必要的Add(),Remove()等方法,类结构图如下::阳诃MdrfU:vnidIKEntivt():voidI曲-

6、心il也山:FictvrGr&pAicsI0:void刑血A;啊死:uoii;ri:iiiil-1.11.:iM仝令乞+甘閃讥:弟+Add(I;viid沁0心:voidetCh:;小川:少:勺Circle+胡的:戦*+k&niD-Me)1voidJ-图4示意性代码:publicabstractclassGraphicsprotectedstring_name;publicGraphics(stringname)this._name=name;publicabstractvoidDraw();publicabstractvoidAdd();publicabstractvoidRemove();p

7、ublicclassPicture:GraphicsprotectedArrayListpicList=newArrayList();publicPicture(stringname):base(name)publicoverridevoidDraw()Console.WriteLine(Drawa+_name.ToString();foreach(GraphicsginpicList)g.Draw();publicoverridevoidAdd(Graphicsg)picList.Add(g);publicoverridevoidRemove(Graphicsg)picList.Remove

8、(g);publicclassLine:GraphicspublicLine(stringname):base(name)publicoverridevoidDraw()Console.WriteLine(Drawa+_name.ToString();publicoverridevoidAdd(Graphicsg)publicoverridevoidRemove(Graphicsg)publicclassCircle:GraphicspublicCircle(stringname):base(name)publicoverridevoidDraw()Console.WriteLine(Draw

9、a+_name.ToString();publicoverridevoidAdd(Graphicsg)publicoverridevoidRemove(Graphicsg)publicclassRectangle:GraphicspublicRectangle(stringname):base(name)publicoverridevoidDraw()Console.WriteLine(Drawa+_name.ToString();publicoverridevoidAdd(Graphicsg)publicoverridevoidRemove(Graphicsg)这样引入Composite模式

10、后,客户端程序不再依赖于复合图像元素的内部实现了。然而,我们程序中仍然存在着问题,因为Line,Rectangle.Circle已经没有了子对象,它是一个基本图像元素,因此Add(),Remove()的方法对于它来说没有任何意义,而且把这种错误不会在编译的时候报错,把错误放在了运行期,我们希望能够捕获到这类错误,并加以处理,稍微改进一下我们的程序:publicclassLine:GraphicspublicLine(stringname):base(name)publicoverridevoidDraw()Console.WriteLine(Drawa+_name.ToString();pub

11、licoverridevoidAdd(Graphicsg)/抛出一个我们自定义的异常publicoverridevoidRemove(Graphicsg)/抛出一个我们自定义的异常这样改进以后,我们可以捕获可能出现的错误,做进一步的处理。上面的这种实现方法属于透明式的Composite模式,如果我们想要更安全的一种做法,就需要把管理子对象的方法声明在树枝构件Picture类里面,这样如果叶子节点Line,Rectangle,Circle使用这些方法时,在编译期就会出错,看一下类结构图:图5示意性代码:publicabstractclassGraphicsprotectedstring_name

12、;publicGraphics(stringname)this._name=name;publicabstractvoidDraw();publicclassPicture:GraphicsprotectedArrayListpicList=newArrayList();publicPicture(stringname):base(name)publicoverridevoidDraw()Console.WriteLine(Drawa+_name.ToString();foreach(GraphicsginpicList)g.Draw();publicvoidAdd(Graphicsg)pic

13、List.Add(g);publicvoidRemove(Graphicsg)picList.Remove(g);publicclassLine:GraphicspublicLine(stringname):base(name)publicoverridevoidDraw()Console.WriteLine(Drawa+_name.ToString();publicclassCircle:GraphicspublicCircle(stringname):base(name)publicoverridevoidDraw()Console.WriteLine(Drawa+_name.ToStri

14、ng();publicclassRectangle:GraphicspublicRectangle(stringname):base(name)publicoverridevoidDraw()Console.WriteLine(Drawa+_name.ToString();这种方式属于安全式的Composite模式,在这种方式下,虽然避免了前面所讨论的错误,但是它也使得叶子节点和树枝构件具有不一样的接口。这种方式和透明式的Composite各有优劣,具体使用哪一个,需要根据问题的实际情况而定。通过Composite模式,客户程序在调用Draw()的时候不用再去判断复杂图像元素中的子对象到底是基

15、本图像元素,还是复杂图像元素,看一下简单的客户端调用:publicclassApppublicstaticvoidMain()Pictureroot=newPicture(Root);root.Add(newLine(Line);root.Add(newCircle(Circle);Rectangler=newRectangle(Rectangle);root.Add(r);root.Draw();NET中的组合模式如果有人用过EnterpriseLibrary2.0,定在源程序中看到了一个叫做ObjectBuilder的程序集,顾名思义,它是用来负责对象的创建工作的,而在ObjectBuil

16、der中,有一个被称为定位器的东西,通过定位器,可以很容易的找到对象,它的结构采用链表结构,每一个节点是一个键值对,用来标识对象的唯一性,使得对象不会被重复创建。定位器的链表结构采用可枚举的接口类来实现,这样我们可以通过一个迭代器来遍历这个链表。同时多个定位器也被串成一个链表。具体地说就是多个定位器组成一个链表,表中的每一个节点是一个定位器,定位器本身又是一个链表,表中保存着多个由键值对组成的对象的节点。所以这是一个典型的Composite模式的例子,来看它的结构图:图6正如我们在图中所看到的,IReadableLocator定义了最上层的定位器接口方法,它基本上具备了定位器的大部分功能。部分

17、代码:publicinterfaceIReadableLocator:IEnumerableKeyValuePair/返回定位器中节点的数量intCountget;/一个指向父节点的引用IReadableLocatorParentLocatorget;/表示定位器是否只读boolReadOnlyget;/查询定位器中是否已经存在指定键值的对象boolContains(objectkey);/查询定位器中是否已经存在指定键值的对象,根据给出的搜索选项,表示是否要向上回溯继续寻找。boolContains(objectkey,SearchModeoptions);/使用谓词操作来查找包含给定对象的

18、定位器IReadableLocatorFindBy(PredicateKeyValuePairpredicate);/根据是否回溯的选项,使用谓词操作来查找包含对象的定位器IReadableLocatorFindBy(SearchModeoptions,PredicateKeyValuePairpredicate);/从定位器中获取一个指定类型的对象TItemGet();/从定位其中获取一个指定键值的对象TItemGet(objectkey);/根据选项条件,从定位其中获取一个指定类型的对象TItemGet(objectkey,SearchModeoptions);/给定对象键值获取对象的非泛

19、型重载方法objectGet(objectkey);/给定对象键值带搜索条件的非泛型重载方法objectGet(objectkey,SearchModeoptions);一个抽象基类ReadableLocator用来实现这个接口的公共方法。两个主要的方法实现代码如下:publicabstractclassReadableLocator:IReadableLocator/*/查找定位器,最后返回一个只读定位器的实例/publicIReadableLocatorFindBy(SearchModeoptions,PredicateKeyValuePairpredicate)if(predicate=

20、null)thrownewArgumentNullException(predicate);if(!Enum.IsDefined(typeof(SearchMode),options)thrownewArgumentException(Properties.Resources.InvalidEnumerationValue,options);Locatorresults=newLocator();IReadableLocatorcurrentLocator=this;while(currentLocator!=null)FindInLocator(predicate,results,curre

21、ntLocator);currentLocator=options=SearchMode.Local?null:currentLocator.ParentLocator;returnnewReadOnlyLocator(results);/*/遍历定位器/privatevoidFindInLocator(PredicateKeyValuePairpredicate,Locatorresults,IReadableLocatorcurrentLocator)foreach(KeyValuePairkvpincurrentLocator)if(!results.Contains(kvp.Key)&

22、predicate(kvp)results.Add(kvp.Key,kvp.Value);可以看到,在FindBy方法里面,循环调用了FindlnLocator方法,如果查询选项是只查找当前定位器,那么循环终止,否则沿着定位器的父定位器继续向上查找。FindlnLocator方法就是遍历定位器,然后把找到的对象存入一个临时的定位器。最后返回一个只读定位器的新的实例。从这个抽象基类中派生出一个具体类和一个抽象类,一个具体类是只读定位器(ReadOn“Locator),只读定位器实现抽象基类没有实现的方法,它封装了一个实现了IReadableLocator接口的定位器,然后屏蔽内部定位器的写入接口方法。另一个继承的是读写定位器抽象类ReadWriteLocator,为了实现对定位器的写入和删除,这里定义了一个对IReadableLocator接口扩展的接口叫做IReadWriteLocator,在这个接口里面提供了实现定位器的操作:图7实现代码如下:publicinterfaceIReadWriteLocator:IReadableLocator/保存对象到定位器voidAdd(objectkey,objectvalue);/从定位器中删除一个对象,如果成功返

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

评论

0/150

提交评论