设计模式C++实现(4)——外观模式、组合模式

2024-06-18 16:32

本文主要是介绍设计模式C++实现(4)——外观模式、组合模式,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!


软件领域中的设计模式为开发人员提供了一种使用专家设计经验的有效途径。设计模式中运用了面向对象编程语言的重要特性:封装、继承、多态,真正领悟设计模式的精髓是可能一个漫长的过程,需要大量实践经验的积累。最近看设计模式的书,对于每个模式,用C++写了个小例子,加深一下理解。主要参考《大话设计模式》和《设计模式:可复用面向对象软件的基础》(DP)两本书。本文介绍外观模式和组合模式的实现。

       外观模式应该是用的很多的一种模式,特别是当一个系统很复杂时,系统提供给客户的是一个简单的对外接口,而把里面复杂的结构都封装了起来。客户只需使用这些简单接口就能使用这个系统,而不需要关注内部复杂的结构。DP一书的定义:为子系统中的一组接口提供一个一致的界面, 外观模式定义了一个高层接口,这个接口使得这一子系统更加容易使用。举个编译器的例子,假设编译一个程序需要经过四个步骤:词法分析、语法分析、中间代码生成、机器码生成。学过编译都知道,每一步都很复杂。对于编译器这个系统,就可以使用外观模式。可以定义一个高层接口,比如名为Compiler的类,里面有一个名为Run的函数。客户只需调用这个函数就可以编译程序,至于Run函数内部的具体操作,客户无需知道。下面给出UML图,以编译器为实例。


        相应的代码实现为:

[cpp]  view plain  copy
  1. class Scanner  
  2. {  
  3. public:  
  4.     void Scan() { cout<<"词法分析"<<endl; }  
  5. };  
  6. class Parser  
  7. {  
  8. public:  
  9.     void Parse() { cout<<"语法分析"<<endl; }  
  10. };  
  11. class GenMidCode  
  12. {  
  13. public:  
  14.     void GenCode() { cout<<"产生中间代码"<<endl; }  
  15. };  
  16. class GenMachineCode  
  17. {  
  18. public:  
  19.     void GenCode() { cout<<"产生机器码"<<endl;}  
  20. };  
  21. //高层接口  
  22. class Compiler  
  23. {  
  24. public:  
  25.     void Run()   
  26.     {  
  27.         Scanner scanner;  
  28.         Parser parser;  
  29.         GenMidCode genMidCode;  
  30.         GenMachineCode genMacCode;  
  31.         scanner.Scan();  
  32.         parser.Parse();  
  33.         genMidCode.GenCode();  
  34.         genMacCode.GenCode();  
  35.     }  
  36. };  

       客户使用方式:

[cpp]  view plain  copy
  1. int main()  
  2. {  
  3.     Compiler compiler;  
  4.     compiler.Run();  
  5.     return 0;  
  6. }  
       这就是外观模式,它有几个特点(摘自DP一书),(1)它对客户屏蔽子系统组件,因而减少了客户处理的对象的数目并使得子系统使用起来更加方便。(2)它实现了子系统与客户之间的松耦合关系,而子系统内部的功能组件往往是紧耦合的。(3)如果应用需要,它并不限制它们使用子系统类。

       结合上面编译器这个例子,进一步说明。对于(1),编译器类对客户屏蔽了子系统组件,客户只需处理编译器的对象就可以方便的使用子系统。对于(2),子系统的变化,不会影响到客户的使用,体现了子系统与客户的松耦合关系。对于(3),如果客户希望使用词法分析器,只需定义词法分析的类对象即可,并不受到限制。

      外观模式在构建大型系统时非常有用。接下来介绍另一种模式,称为组合模式。感觉有点像外观模式,刚才我们实现外观模式时,在Compiler这个类中包含了多个类的对象,就像把这些类组合在了一起。组合模式是不是这个意思,有点相似,其实不然。

      DP书上给出的定义:将对象组合成树形结构以表示“部分-整体”的层次结构。组合使得用户对单个对象和组合对象的使用具有一致性。注意两个字“树形”。这种树形结构在现实生活中随处可见,比如一个集团公司,它有一个母公司,下设很多家子公司。不管是母公司还是子公司,都有各自直属的财务部、人力资源部、销售部等。对于母公司来说,不论是子公司,还是直属的财务部、人力资源部,都是它的部门。整个公司的部门拓扑图就是一个树形结构。

      下面给出组合模式的UML图。从图中可以看到,FinanceDepartment、HRDepartment两个类作为叶结点,因此没有定义添加函数。而ConcreteCompany类可以作为中间结点,所以可以有添加函数。那么怎么添加呢?这个类中定义了一个链表,用来放添加的元素。

       相应的代码实现为:

[cpp]  view plain  copy
  1. class Company    
  2. {  
  3. public:  
  4.     Company(string name) { m_name = name; }  
  5.     virtual ~Company(){}  
  6.     virtual void Add(Company *pCom){}  
  7.     virtual void Show(int depth) {}  
  8. protected:  
  9.     string m_name;  
  10. };  
  11. //具体公司  
  12. class ConcreteCompany : public Company    
  13. {  
  14. public:  
  15.     ConcreteCompany(string name): Company(name) {}  
  16.     virtual ~ConcreteCompany() {}  
  17.     void Add(Company *pCom) { m_listCompany.push_back(pCom); } //位于树的中间,可以增加子树  
  18.     void Show(int depth)  
  19.     {  
  20.         for(int i = 0;i < depth; i++)  
  21.             cout<<"-";  
  22.         cout<<m_name<<endl;  
  23.         list<Company *>::iterator iter=m_listCompany.begin();  
  24.         for(; iter != m_listCompany.end(); iter++) //显示下层结点  
  25.             (*iter)->Show(depth + 2);  
  26.     }  
  27. private:  
  28.     list<Company *> m_listCompany;  
  29. };  
  30. //具体的部门,财务部  
  31. class FinanceDepartment : public Company   
  32. {  
  33. public:  
  34.     FinanceDepartment(string name):Company(name){}  
  35.     virtual ~FinanceDepartment() {}  
  36.     virtual void Show(int depth) //只需显示,无限添加函数,因为已是叶结点  
  37.     {  
  38.         for(int i = 0; i < depth; i++)  
  39.             cout<<"-";  
  40.         cout<<m_name<<endl;  
  41.     }  
  42. };  
  43. //具体的部门,人力资源部  
  44. class HRDepartment :public Company    
  45. {  
  46. public:  
  47.     HRDepartment(string name):Company(name){}  
  48.     virtual ~HRDepartment() {}  
  49.     virtual void Show(int depth) //只需显示,无限添加函数,因为已是叶结点  
  50.     {  
  51.         for(int i = 0; i < depth; i++)  
  52.             cout<<"-";  
  53.         cout<<m_name<<endl;  
  54.     }  
  55. };  

         客户使用方式:

[cpp]  view plain  copy
  1. int main()  
  2. {  
  3.     Company *root = new ConcreteCompany("总公司");  
  4.     Company *leaf1=new FinanceDepartment("财务部");  
  5.     Company *leaf2=new HRDepartment("人力资源部");  
  6.     root->Add(leaf1);  
  7.     root->Add(leaf2);  
  8.   
  9.     //分公司A  
  10.     Company *mid1 = new ConcreteCompany("分公司A");  
  11.     Company *leaf3=new FinanceDepartment("财务部");  
  12.     Company *leaf4=new HRDepartment("人力资源部");  
  13.     mid1->Add(leaf3);  
  14.     mid1->Add(leaf4);  
  15.     root->Add(mid1);  
  16.     //分公司B  
  17.     Company *mid2=new ConcreteCompany("分公司B");  
  18.     FinanceDepartment *leaf5=new FinanceDepartment("财务部");  
  19.     HRDepartment *leaf6=new HRDepartment("人力资源部");  
  20.     mid2->Add(leaf5);  
  21.     mid2->Add(leaf6);  
  22.     root->Add(mid2);  
  23.     root->Show(0);  
  24.   
  25.     delete leaf1; delete leaf2;  
  26.     delete leaf3; delete leaf4;  
  27.     delete leaf5; delete leaf6;   
  28.     delete mid1; delete mid2;  
  29.     delete root;  
  30.     return 0;  
  31. }  

        上面的实现方式有缺点,就是内存的释放不好,需要客户自己动手,非常不方便。有待改进,比较好的做法是让ConcreteCompany类来释放。因为所有的指针都是存在ConcreteCompany类的链表中。C++的麻烦,没有垃圾回收机制。

















本文转自:

http://blog.csdn.net/wuzhekai1985/article/details/6667564










这篇关于设计模式C++实现(4)——外观模式、组合模式的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1072514

相关文章

Java实现时间与字符串互相转换详解

《Java实现时间与字符串互相转换详解》这篇文章主要为大家详细介绍了Java中实现时间与字符串互相转换的相关方法,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录一、日期格式化为字符串(一)使用预定义格式(二)自定义格式二、字符串解析为日期(一)解析ISO格式字符串(二)解析自定义

opencv图像处理之指纹验证的实现

《opencv图像处理之指纹验证的实现》本文主要介绍了opencv图像处理之指纹验证的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学... 目录一、简介二、具体案例实现1. 图像显示函数2. 指纹验证函数3. 主函数4、运行结果三、总结一、

Springboot处理跨域的实现方式(附Demo)

《Springboot处理跨域的实现方式(附Demo)》:本文主要介绍Springboot处理跨域的实现方式(附Demo),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不... 目录Springboot处理跨域的方式1. 基本知识2. @CrossOrigin3. 全局跨域设置4.

Spring Boot 3.4.3 基于 Spring WebFlux 实现 SSE 功能(代码示例)

《SpringBoot3.4.3基于SpringWebFlux实现SSE功能(代码示例)》SpringBoot3.4.3结合SpringWebFlux实现SSE功能,为实时数据推送提供... 目录1. SSE 简介1.1 什么是 SSE?1.2 SSE 的优点1.3 适用场景2. Spring WebFlu

基于SpringBoot实现文件秒传功能

《基于SpringBoot实现文件秒传功能》在开发Web应用时,文件上传是一个常见需求,然而,当用户需要上传大文件或相同文件多次时,会造成带宽浪费和服务器存储冗余,此时可以使用文件秒传技术通过识别重复... 目录前言文件秒传原理代码实现1. 创建项目基础结构2. 创建上传存储代码3. 创建Result类4.

SpringBoot日志配置SLF4J和Logback的方法实现

《SpringBoot日志配置SLF4J和Logback的方法实现》日志记录是不可或缺的一部分,本文主要介绍了SpringBoot日志配置SLF4J和Logback的方法实现,文中通过示例代码介绍的非... 目录一、前言二、案例一:初识日志三、案例二:使用Lombok输出日志四、案例三:配置Logback一

Python如何使用__slots__实现节省内存和性能优化

《Python如何使用__slots__实现节省内存和性能优化》你有想过,一个小小的__slots__能让你的Python类内存消耗直接减半吗,没错,今天咱们要聊的就是这个让人眼前一亮的技巧,感兴趣的... 目录背景:内存吃得满满的类__slots__:你的内存管理小助手举个大概的例子:看看效果如何?1.

Python+PyQt5实现多屏幕协同播放功能

《Python+PyQt5实现多屏幕协同播放功能》在现代会议展示、数字广告、展览展示等场景中,多屏幕协同播放已成为刚需,下面我们就来看看如何利用Python和PyQt5开发一套功能强大的跨屏播控系统吧... 目录一、项目概述:突破传统播放限制二、核心技术解析2.1 多屏管理机制2.2 播放引擎设计2.3 专

Python实现无痛修改第三方库源码的方法详解

《Python实现无痛修改第三方库源码的方法详解》很多时候,我们下载的第三方库是不会有需求不满足的情况,但也有极少的情况,第三方库没有兼顾到需求,本文将介绍几个修改源码的操作,大家可以根据需求进行选择... 目录需求不符合模拟示例 1. 修改源文件2. 继承修改3. 猴子补丁4. 追踪局部变量需求不符合很

idea中创建新类时自动添加注释的实现

《idea中创建新类时自动添加注释的实现》在每次使用idea创建一个新类时,过了一段时间发现看不懂这个类是用来干嘛的,为了解决这个问题,我们可以设置在创建一个新类时自动添加注释,帮助我们理解这个类的用... 目录前言:详细操作:步骤一:点击上方的 文件(File),点击&nbmyHIgsp;设置(Setti