【M26】限制某个class所能产生的对象数量
1、每当产生一个对象,必定调用构造方法。因此,禁止产生对象的做法就是,将所有的构造方法声明为private。
2、只有在类的内部才可以访问private成员,有两层含义:在类的内部可以访问this的private成员,同时可以访问同类对象的private成员。
3、将构造方法声明为private,只是限制了在外部调用构造方法产生对象,还是有办法可以产生对象。办法有:
a、类暴露一个static方法,在static方法内部调用private构造方法,产生对象返回。
b、在类中声明友元方法,或者友元类,这样的话,就可以访问该类的private构造方法。
4、考虑,只产生一个对象。该怎么办?使用友元方法,内部有static对象,返回static对象的引用。比如:只产生一个打印机。
在类中声明:friend Printer& thePrinter();
实现为:
printer& thePrinter()
{
static Printer p;
return p;
}
5、上面的方式有个问题,友元方法是个全局方法,意味着放大了作用域。有没有更好的办法呢?
a、第一个办法是,使用命名空间,将与Printer有关的内容放在这个命名空间内,这样就缩小了作用域。
b、第二个方法是,让类暴露一个static方法。如下:
在类中声明:static Printer& thePrinter();
实现为:
Printer& Printer::thePrinter()
{
static Printer p;
return p;
}
从封装的角度看,类本身表示一个范围,可以认为命名空间的意思。只不过,命名空间内的东西都是public,而类中可以细化为public,protected,private。
6、现在考虑,方法中的static对象。
a、首先一点,对于方法中的static对象,也就是local static对象,方法不被调用,static对象就绝不会产生。而对于non-local static对象(包括全局对象,命名空间内的对象,file作用域,class中的static对象),即使从未被调用,也要构造析构对象。从c++的角度看,不要为不使用的东西产生代价。因此,使用local static对象更好。
b、还有第二点,在一个编译单元内的static对象,C++保证按声明的顺序初始化,而对于不同编译单元内的static对象,顺序是不确定的,因此,程序员不能依赖某种初始化顺序。而对于,方法中的static对象,确定是第一次调用方法的时候,初始化static对象。
c、还有一点,这个方法不能声明为inline,思考为什么?这个方法中,使用static,目的就是,返回的引用都是这个static对象。假如使用inline,可认为编译器在每个调用的地方进行文本替换,这样的话,就会产生多个static对象,这明显不是我们所期望的。
7、现在考虑,更一般化的问题。比如,我们有5个打印机,如何限制产生的对象个数不能大于5?
在类中维护两个字段:已经产生的对象个数NumObjects和允许最多的对象个数MaxObjects,这两个字段对类和所有对象有意义,因此是static。每次构造对象的时候,检查NumObjects<MaxObjects,条件满足,++NumObjects,构造一个新对象返回。条件不满足,抛出一个异常。每次析构对象的时候,--NumObjects。
8、上面的方式,在调用构造方法的时候,对NumObjects累加。这将导致下面的两个问题:
a、考虑,ColorPrinter继承Printer,实例化ColorPrinter对象的时候,会调用Printer 的构造方法,对NumObjects累加,这不是我们所期望的。
b、考虑类A中内含Printer对象,A a1,a2; 导致调用Printer构造方法两次,这也不是我们所期望的。
9、怎么解决上面的问题? 问题的关键是Printer对象在3种状态下生存:a、它自己;b、子类的父类成分中;c、内嵌于其它对象中。因此,解决的办法是,禁止后面的两种状态。也就是将构造方法声明为private。 构造方法声明为private,不能被继承,同时不能在外部调用构造方法构造对象。但同时,又必须产生对象给外部用,也就是上面的方法:类暴露static方法,或者类中声明友元方法或者友元类。
10、考虑下面的问题,如果有很多类似Printer的类,该怎么办呢?
把共用的代码放到父类中,父类负责管理对象的个数,因为可以对不同的类封装,因此父类是模板类。如下:
class Printer: private Counted<Printer>
【M26】限制某个class所能产生的对象数量的更多相关文章
- 2-移远GSM/GPRS M26 模块 Mini板 开发板(M26入门)
https://www.cnblogs.com/yangfengwu/p/11214553.html 资料获取,首先说一下....很多东西需要自己悟,没有QQ群,没有微信群,论坛也寥寥无几!!!! 估 ...
- 1-移远GSM/GPRS M26 模块 Mini板 开发板(使用说明)
板子预览 引脚说明 供电 关于串口电压匹配引脚: 上面一版朋友测试反应的问题 (上面的内容不删除,因为已经出售了1套) 1,源码开发完以后,烧录完成 PWRKEY按键不能使用了,需要断电上电,那么就需 ...
- ListView初探
一.ListView介绍 在Android开发中ListView是比较常用的控件,常用于以列表的形式显示数据集及根据数据的长度自适应显示. ListView通常有两个主要功能点: (1)将数据集填充到 ...
- CSS字体记录
aaarticlea/png;base64,iVBORw0KGgoAAAANSUhEUgAAAaYAAACnCAIAAADVOG9FAAAgAElEQVR4nOy9eXwcxZk/vL/9ve8eb7 ...
- BOOST.Asio——Tutorial
=================================版权声明================================= 版权声明:原创文章 谢绝转载 啥说的,鄙视那些无视版权随 ...
- iOS GCD NSOperation NSThread等多线程各种举例详解(拷贝)
2年多的iOS之路匆匆而过,期间也拜读来不少大神的博客,近来突然为自己一直做伸手党感到羞耻,是时候回馈社会.回想当年自己还是小白的时候,照着一些iOS多线程教程学,也只是照抄,只知其然.不知其所以然. ...
- SQL Server优化技巧之SQL Server中的"MapReduce"
日常的OLTP环境中,有时会涉及到一些统计方面的SQL语句,这些语句可能消耗巨大,进而影响整体运行环境,这里我为大家介绍如何利用SQL Server中的”类MapReduce”方式,在特定的统计情形中 ...
- Spring 框架的设计理念与设计模式分析
转载地址:https://www.ibm.com/developerworks/cn/java/j-lo-spring-principle/ Spring 作为现在最优秀的框架之一,已被广泛的使用,并 ...
- 【js类库Raphaël】基于svg中的path画40%表示的环型图
一.可供参考的文档资料. raphaeljs官网:http://raphaeljs.com/ w3c关于path的介绍:http://www.w3.org/TR/2003/REC-SVG11-200 ...
随机推荐
- 【转】个人常用iOS第三方库以及XCode插件介绍 -- 不错
原文网址:http://adad184.com/2015/07/08/my-favorite-libraries-and-plugins/ 第三方库是现在的程序员离不开的东西 不光是APP开发 基本上 ...
- awk简单使用『摘.非原创』
1 awk简介 awk是一种编程语言,用于在linux/unix下对文本和数据进行处理.数据可以来自标准输入.一个或多个文件,或其它命令的输出.它支持用户自定义函数和动态正则表达式等先进功能,是lin ...
- hdu 3336 count the string(KMP+dp)
题意: 求给定字符串,包含的其前缀的数量. 分析: 就是求所有前缀在字符串出现的次数的和,可以用KMP的性质,以j结尾的串包含的串的数量,就是next[j]结尾串包含前缀的数量再加上自身是前缀,dp[ ...
- HDU5715 XOR 游戏 二分+字典树+dp
当时Astar复赛的时候只做出1题,赛后补题(很长时间后才补,懒真是要命),发现这是第二简单的 分析: 这个题,可以每次二分区间的最小异或和 进行check的时候用dp进行判断,dp[i][j]代表前 ...
- HDU-4035 Maze
http://acm.hdu.edu.cn/showproblem.php?pid=4035 树上的概率dp. Maze Time Limit: 2000/1000 MS (Java/Others ...
- IOS AsyncSocket
导入AsyncSocket.h AsyncSocket.m AsyncUdpSocket.h AsyncUdpSocket.m 以及 CFNetWork.framework async ...
- 《Python CookBook2》 第一章 文本 - 控制大小写 && 访问子字符串
控制大小写 任务: 将一个字符串由大写转成小写,或者泛起到而行之. 解决方案: >>> a = 'a'.upper() >>> a 'A' >>> ...
- 修改Zabbix默认运行账户
默认Zabbix运行的账户是Zabbix,但在自动部署的时候,Agent与Server的先后顺序不定,而且官方不建议两者使用同一个账户. 所以,解压压缩包后,进入目录: vi configure ...
- JQuery插件的学习
此前一直想就关于Jquery插件的开发,做一个深入的学习,由于各种原因,当然主要是自己太懒了...今天就系统分析一下Jquery插件的开发(参考了http://www.xprogrammer.com/ ...
- 判断相同区间(lazy) 多校8 HDU 5828 Rikka with Sequence
// 判断相同区间(lazy) 多校8 HDU 5828 Rikka with Sequence // 题意:三种操作,1增加值,2开根,3求和 // 思路:这题与HDU 4027 和HDU 5634 ...