设计模式C++实现(2)——单例模式

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

      单例的一般实现比较简单,下面是代码和UML图。由于构造函数是私有的,因此无法通过构造函数实例化,唯一的方法就是通过调用静态函数GetInstance。

      UML图:

      代码:

[cpp] view plaincopyprint?
  1. //Singleton.h  
  2. class Singleton    
  3. {  
  4. public:  
  5.     static Singleton* GetInstance();  
  6. private:  
  7.     Singleton() {}  
  8.     static Singleton *singleton;  
  9. };  
  10. //Singleton.cpp  
  11. Singleton* Singleton::singleton = NULL;  
  12. Singleton* Singleton::GetInstance()  
  13. {  
  14.     if(singleton == NULL)  
  15.         singleton = new Singleton();  
  16.     return singleton;  
  17. }  

       这里只有一个类,如何实现Singleton类的子类呢?也就说Singleton有很多子类,在一种应用中,只选择其中的一个。最容易就是在GetInstance函数中做判断,比如可以传递一个字符串,根据字符串的内容创建相应的子类实例。这也是DP书上的一种解法,书上给的代码不全。这里重新实现了一下,发现不是想象中的那么简单,最后实现的版本看上去很怪异。在VS2008下测试通过。

[cpp] view plaincopyprint?
  1. //Singleton.h  
  2. #pragma once  
  3. #include <iostream>  
  4. using namespace std;  
  5.   
  6. class Singleton    
  7. {  
  8. public:  
  9.     static Singleton* GetInstance(const char* name);  
  10.     virtual void Show() {}  
  11. protected//必须为保护,如果是私有属性,子类无法访问父类的构造函数  
  12.     Singleton() {}  
  13. private:  
  14.     static Singleton *singleton; //唯一实例的指针  
  15. };  
  16.   
  17. //Singleton.cpp  
  18. #include "Singleton.h"  
  19. #include "SingletonA.h"  
  20. #include "SingletonB.h"  
  21. Singleton* Singleton::singleton = NULL;  
  22. Singleton* Singleton::GetInstance(const char* name)  
  23. {  
  24.     if(singleton == NULL)  
  25.     {  
  26.         if(strcmp(name, "SingletonA") == 0)  
  27.             singleton = new SingletonA();  
  28.         else if(strcmp(name,"SingletonB") == 0)  
  29.             singleton = new SingletonB();  
  30.         else   
  31.             singleton = new Singleton();  
  32.     }  
  33.     return singleton;  
  34. }  
[cpp] view plaincopyprint?
  1. //SingletonA.h  
  2. #pragma once  
  3. #include "Singleton.h"  
  4. class SingletonA: public Singleton  
  5. {  
  6.     friend class Singleton; //必须为友元类,否则父类无法访问子类的构造函数  
  7. public:  
  8.     void Show() { cout<<"SingletonA"<<endl; }  
  9. private:   //为保护属性,这样外界无法通过构造函数进行实例化  
  10.     SingletonA() {}   
  11. };  
  12. //SingletonB.h  
  13. #pragma once  
  14. #include "Singleton.h"  
  15. class SingletonB: public Singleton  
  16. {  
  17.     friend class Singleton; //必须为友元类,否则父类无法访问子类的构造函数  
  18. public:  
  19.     void Show(){ cout<<"SingletonB"<<endl; }  
  20. private:  //为保护属性,这样外界无法通过构造函数进行实例化  
  21.     SingletonB() {}  
  22. };  
[cpp] view plaincopyprint?
  1. #include "Singleton.h"  
  2. int main()  
  3. {  
  4.     Singleton *st = Singleton::GetInstance("SingletonA");  
  5.     st->Show();  
  6.     return 0;  
  7. }  

       上面代码有一个地方很诡异,父类为子类的友元,如果不是友元,函数GetInstance会报错,意思就是无法调用SingletonA和SIngletonB的构造函数。父类中调用子类的构造函数,我还是第一次碰到。当然了把SingletonA和SIngletonB的属性设为public,GetInstance函数就不会报错了,但是这样外界就可以定义这些类的对象,违反了单例模式。
       看似奇怪,其实也容易解释。在父类中构建子类的对象,相当于是外界调用子类的构造函数,因此当子类构造函数的属性为私有或保护时,父类无法访问。为共有时,外界就可以访问子类的构造函数了,此时父类当然也能访问了。只不过为了保证单例模式,所以子类的构造函数不能为共有,但是又希望在父类中构造子类的对象,即需要调用子类的构造函数,这里没有办法才出此下策:将父类声明为子类的友元类。

      本人享有博客文章的版权,转载请标明出处 http://blog.csdn.net/wuzhekai1985

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/436143.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

php聊天室技术,PHP聊天室技术

PHP聊天室技术推荐查看本文HTML版本1&#xff0e;前言上网聊天是时下最流行的交友方式。各大网站推出的聊天室都各具特色。聊天室主要分为WebChat、BBSChat两种。BBSChat是基于Telnet的Tcp协议&#xff0c;是BBS的附设功能&#xff0c;需要客户端Telnet程序。WebChat则采用…

【转】深入理解Windows消息机制

转自&#xff1a;https://blog.csdn.net/liulianglin/article/details/14449577 今天我们来学一学Windows消息机制&#xff0c;我们知道在传统的C语音程序中&#xff0c;当我们需要打开一个文件时&#xff0c;我们可以调用fopen()函数&#xff0c;这个函数最后又会调用操作系统…

System.Drawing.Color.FromArgb(144,238,255);

if(dg.Items[i].Cells[iRow].Text "没完成") { dg.Items[i].BackColorSystem.Drawing.Color.FromArgb(144,238,255); } if(dg.Items[i].Cells[iRow].Text "已完成") { dg.Items[i].BackColorSystem.Drawing.Color.LightGreen;…

设计模式C++实现(3)——建造者模式

软件领域中的设计模式为开发人员提供了一种使用专家设计经验的有效途径。设计模式中运用了面向对象编程语言的重要特性&#xff1a;封装、继承、多态&#xff0c;真正领悟设计模式的精髓是可能一个漫长的过程&#xff0c;需要大量实践经验的积累。最近看设计模式的书&#xff0…

php tp 模板assign,thinkphp中怎么让assign在另一个模板里使用呢?

比如现在我有a.html和名字为b的控制器,现在我在b控制器里assing(‘b’,$b),那么请问在a.html中怎么调用b控制器中的assign呢?回复讨论(解决方案)代码是这样的Controller:public function a(){$this->display();}public function b(){if(IS_AJAX){$imageD(‘Image’);$b$ima…

【转】详解Windows消息分类以及WM_COMMAND与WM_NOTIFY的区别,以及模拟发送控件通知消息

转自&#xff1a;http://blog.sina.com.cn/s/blog_4b3c1f950100nten.html Windows消息的分类 1. 标准消息&#xff08;队列消息&#xff09; 除WM_COMMAND之外&#xff0c;所有以WM_开头的消息都是标准消息&#xff0c;如WM_MOUSEMOVE、WM_LBUTTONUP、WM_KEYDOWN、WM_CHAR。…

宽字符串忽略大小写比较的实现(原)

宽字符串忽略大小写比较的实现(原) 孙文涛 2008-07-24 在Mac机器平台上没有wcsicmp 或 wcscasecmp之类的函数实现对宽字符忽略大小写的比较&#xff0c;所以实现了好几种方法。 一个自然的思路是: (1) wcscpy 原字符串到tmp字符串; (2) tolower tmp字符串; (3) 然后调用仅存的w…

有关JAVA考试中数据库的题,javaee期末考试题库,用javaEE编写一个题库系统,要怎么做...

javaEE数据库简单问题。你插入2113数据库的时候是把5261它封装为一4102个对象插入的吗&#xff1f;1653如果封装为一个User对象版的权话(User对象有ID和LEVEL两个属性)ResultSet rs ps.executeQuery();User user new User();if(rs.hasNext()){user rs.next();}关于javaee 中j…

wince下获取mac地址的简单方法!

下&#xff0c;可以通过访问注册表获取mac地址&#xff0c;可是非常可惜的是有些系统的注册表不提供这个键值&#xff0c;另外也可以通过 DeviceIoControl这类函数获得&#xff0c;但是所有方法要么不全面&#xff0c;要么不够简单或者有些平台bsp包根本就不支持&#xff0c;我…

【转】虚拟键码

虚拟键码保存在WM_KEYDOWN、WM_KEYUP、WM_SYSKEYDOWN和WM_SYSKEYUP讯息的wParam参数中。此代码标识按下或释放的键。 中文名 虚拟键码 保存在 WM_KEYDOWN等 程 序 Windows程序 代码标识 按下或释放的键 目录 1 简介2 虚拟键表▪ 完整的虚拟键码表▪ 键盘的扫描码、…

循环赛日程表算法

题目&#xff1a;有n2^k个运动员要进行循环赛。现要设计一个满足以下要求的比赛日程表&#xff1a; &#xff08;1&#xff09;每个选手必须与其他n-1个选手各赛一次 &#xff08;2&#xff09;每个选手一天只能赛一次 &#xff08;3&#xff09;循环赛一共进行n-1天 解题思路&…

js有没有类似php的sleep函数,JavaScript-jQuery有没有类似sleep方法?

cronRun.sleep function(n) {for (var i0,jn*1000;i}}小飞写的function sleep(n) {var start new Date().getTime();while(true) if(new Date().getTime()-start > n) break;}貌似 Date new 的过多啊不过我在我本本上测的 好像不怎么管用&#xff0c;不知道为什么 是不是和…

GetAdaptersInfo获取MAC地址

源代码&#xff1a;#include<atlbase.h>#include<atlconv.h>#include"iphlpapi.h"#pragma comment(lib,"Iphlpapi.lib")int main(int argc,char*argv[]){PIP_ADAPTER_INFO pAdapterInfo;PIP_ADAPTER_INFO pAdapterNULL;DWORD dwRetVal0;pAdapt…

【转】Windows 窗体消息大全(速查)

阅读目录 通用窗口消息滚动条消息按钮控件消息按钮控件通知消息组合框控件消息组合框控件通知消息列表框控件消息列表框控件通知消息DDE消息(Dde.h)工具提示控件消息Rich Edit控件消息(带格式编辑控件)Listview控件消息Listview控件通知消息树控件消息树控件通知消息Header Co…

整数表示

在计算机中&#xff0c;不管程序还是数据都表示为二进制格式。对于我们平常编程常使用的整数有两种类型&#xff0c;一种是无符号整数&#xff0c;另外一种是有符号整数。 相应地&#xff0c;就有两种整数的二进制表达方式。一般我们使用的计算机都是32位,因此对两种整数的二进…

char 转wchar_t 及wchar_t转char

利用widechartomultibyte来转换的函数 通常适合于window平台上使用 #include <tchar.h>#include <windows.h>int _tmain(int argc, _tchar* argv[]){wchar_t pwstr[] l"我是中国人";wchar_t pwstr2[20];char *pcstr (char *)malloc(sizeof(char)*(2 * w…

oracle 12c sql图形化,Oracle 12c PL/SQL程序设计终极指南

Oracle 12c PL/SQL程序设计终极指南作者&#xff1a;孙风栋;王澜;郭晓惠出版日期&#xff1a;2015年06月文件大小&#xff1a;11.73M支持设备&#xff1a;&#xffe5;60.00在线试读适用客户端&#xff1a;言商书局iPad/iPhone客户端&#xff1a;下载 Android客户端&#xff1a…

CSS的语法结构

今天又看了一点书&#xff0c;哎我看书的速度可是够慢的。现摘录下来自己也增强下记忆&#xff0c;反正闲着也是闲着o(∩_∩)o...。 CSS的语法结构 CSS的语法结构由三个部分组成&#xff1a;选择符&#xff0c;(Selector)&#xff0c;属性(Porperty)和值(Value)。 使用方法…

WinCE 控制面板的创建

控制面板的组件,其实就是一个地地道道的DLL文件,所不同的是其后缀名为.CPL而已.控制面板组件对外必须要实现一个CPlApplet接口,其原型为:LONG CALLBACK CPlApplet(HWND hwndCPL,UINT message, LPARAM lParam1, LPARAM lParam2).而在此函数之中,为了使组件正常运作,我们必须要处…

oracle 存储同步,Oracle数据库知识——存储过程篇

在线QQ客服&#xff1a;1922638专业的SQL Server、MySQL数据库同步软件存储过程是一组用于完成特定功能的SQL语句&#xff0c;该语句已编译并存储在数据库中。用户通过指定存储过程的名称并提供参数(如果存储过程具有参数)来执行它。存储过程是数据库中的重要对象。任何设计良好…