1、只能在堆上使用的类

栈上对象通过自动调用析构函数释放空间,将该类的析构函数定义为private就可以阻止其被自动调用,从而阻止在栈上使用该类的对象,为了避免资源泄漏问题,在堆上使用该类的对象时我们必须手动调用其析构函数。代码如下。

  1. #include <iostream>
  2. using namespace std;
  3. class CHeapOnly {
  4. public:
  5. CHeapOnly() {
  6. cout << "Constructor of CHeapOnly" << endl;
  7. }
  8. void Destroy() const {
  9. delete this;
  10. }
  11. private:
  12. ~CHeapOnly() {
  13. cout << "Destructor of CHeapOnly" << endl;
  14. }
  15. };
  16. int main() {
  17. CHeapOnly* pHeap = new CHeapOnly;
  18. pHeap->Destroy();
  19. //CHeapOnly objHeap; //error
  20. return 0;
  21. }

2、只能在栈上使用的类

堆上对象的析构需要调用delete operator,delete operator会调用析构函数,然后调用operator delete,operator delete是函数且可以被重载,将operator delete定义为private就可以阻止堆上对象的析构,从而阻止其在堆上使用。类似的,我们可以将operator new定义为private阻止堆上创建该类的对象,因为堆上创建对象需要调用new operator,new operator会调用operator new,然后调用类的构造函数。

  1. #include <iostream>
  2. using namespace std;
  3. class CStackOnly {
  4. public:
  5. CStackOnly() {
  6. cout << "Constructor of CStackOnly" << endl;
  7. }
  8. ~CStackOnly() {
  9. cout << "Destrucotr of CStackOnly" << endl;
  10. }
  11. private:
  12. void* operator new(size_t size) {
  13. }
  14. void operator delete(void * pointee) {
  15. }
  16. };
  17. int main() {
  18. CStackOnly objStack;
  19. //CStackOnly* pStack = new CStackOnly; //error
  20. return 0;
  21. }

3、不能被继承的类

构建派生类对象时会调用基类的构造函数,将一个类的的构造函数定义为private就可以阻止被继承。那么如何生成该类的对象呢?这时可以一个工厂方法。

还有一种方法类似boost::noncopyable,代码如下,我们需要使用虚基类的性质,虚基类的构造是由most derived class负责,具体请参考《深度探索C++对象模型》第210页,书中指出“constructor的函数本体因而必须条件式地测试传进来的参数,然后决定调用或者不调用相关的virtual base class constructors”,也指出“只有当一个完整的class object被定义出来时,它才会被调用;如果object只是某个完整object的subject,它就不会被调用”,换句话说就是most-derived class才会调用虚基类的构造函数。

  1. #include <iostream>
  2. using namespace std;
  3. class CNonderivable;
  4. class CNonderivableHelper
  5. {
  6. private:
  7. friend class CNonderivable;
  8. CNonderivableHelper(){};
  9. };
  10. class CNonderivable:private virtual CNonderivableHelper {
  11. public:
  12. CNonderivable(){};
  13. ~CNonderivable(){};
  14. };
  15. class Derived:public CNonderivable {
  16. };
  17. int main() {
  18. CNonderivable x;
  19. //Derived y; //error
  20. return 0;
  21. }

4、不能执行拷贝或赋值的类

拷贝和赋值需要调用拷贝构造函数和赋值运算符,如果将这两个函数声明为private就可以阻止执行拷贝或赋值,更好的方法是像boost::noncopyable一样,或者使用C++中的delete关键词,然后继承这个类。需要注意的是,将拷贝构造函数和赋值运算符声明为private的,只能阻止用户代码不能拷贝这个类型的对象,但是友元和成员函数仍旧可以拷贝对象,为了阻止友元和成员函数进行拷贝,我们只能声明不能定义它们。

  1. //方法1
  2. class Uncopyable {
  3. public:
  4. Uncopyable() {}
  5. private:
  6. Uncopyable(const Uncopyable&);
  7. Uncopyable& operator=(const Uncopyable&);
  8. };
  9. //方法2
  10. struct NonCopyable {
  11. NonCopyable() = default;
  12. NonCopyable(const NonCopyable&) = delete;
  13. NonCopyable& operator=(const NonCopyable&) = delete;
  14. };
  15. //使用时继承上面两个类中的某一个即可

C++中定义使用受限的类的更多相关文章

  1. java中定义一个CloneUtil 工具类

    其实所有的java对象都可以具备克隆能力,只是因为在基础类Object中被设定成了一个保留方法(protected),要想真正拥有克隆的能力, 就需要实现Cloneable接口,重写clone方法.通 ...

  2. ES6 class类中定义私有变量

    ES6 class类中定义私有变量 class类的不足 看起来, es6 中 class 的出现拉近了 JS 和传统 OOP 语言的距离.但是,它仅仅是一个语法糖罢了,不能实现传统 OOP 语言一样的 ...

  3. JPA 不在 persistence.xml 文件中配置每个Entity实体类的2种解决办法

    在Spring 集成 Hibernate 的JPA方式中,需要在persistence配置文件中定义每一个实体类,这样非常地不方便,远哥目前找到了2种方法.   这2种方式都可以实现不用persist ...

  4. 关于Java中的几种特殊类与接口,及特殊的创建实例的方法

    Java中有一些特殊的类,在教材中讲解的不深,但是确实非常有用的,这里总结一下,里面用到的有网上搜到的内容,这里表示下感谢. 一.成员内部类 成员内部类是在一个内中定义的另外一个类,这个类属于其上的类 ...

  5. IronPython中共享的C#基类如何向下转型

    在项目中,我们使用IronPython来定义工作流脚本来以应对科研多变的需求.项目使用的主要语言仍然是C#,使用C#封装了各种基础服务与基础设施.Python脚本只使用C#提供的服务,或者说只定义了逻 ...

  6. 如何使用其他文件中定义的类Python

    我在文件a.py中定义了一个类class A(object),现在想在b.py中的类B中某个函数中创建一个A的对象,需要如何操作呢? 我在b的头加了import a.py然后使用语句 obj = A( ...

  7. JS中定义类的方法

    JS中定义类的方式有很多种: 1.工厂方式    function Car(){     var ocar = new Object;     ocar.color = "blue" ...

  8. java类中定义接口

    今天看到一个java类中定义了接口,写个备忘录,记录一下 package com.gxf.test; public class Test_interface { public interface sh ...

  9. php中定义类

    <?php class Person{ //定义了一个Person类 public $name; //定义属性name public $age; //定义属性age function __con ...

随机推荐

  1. iomanip头

    #include <iomanip>主要是输入输出控制. Dec 十进制: hex 十六进制: oct 八进制: setw 设置宽度: setfill 设置填充值: setbase 将数字 ...

  2. ecos的mvcl

    m 数据模型抽象层 v 视图 c 控制器 l 业务逻辑 mvc与mvcl区别 mvc中的m是mvcl中m+l

  3. Android中通过pid获取app包名

    String callerPackage = getAppNameByPID(getContext(), Binder.getCallingPid()); private String getAppN ...

  4. TextView 设置图片

    TextView 设置图片 2012-05-17 15:12:38|  分类: Android |  标签:android  textview图片  |举报|字号 订阅     Drawable im ...

  5. Java谜题——库谜题

    1.Java中的不可变对象和可变对象 (1)不可变类:当你获得这个类的实例的引用之后,你不可以改变这个实例的内容.比如:String,BigInteger,BigDecimal,还有基本数据类型的封装 ...

  6. hibernate---一对一双向外键关联 (重要)

    husband--wife: one to one 双向外键关联: 主导方: @OneToOne @JoinColumn(name="wifeId") 被主导方: @OneToOn ...

  7. Java内存回收 - 落日之心的日志 - 网易博客

    body{ font-family: "Microsoft YaHei UI","Microsoft YaHei",SimSun,"Segoe UI& ...

  8. 前台html与后台php通信(上传文件)

    这部分为导入txt文本文件,存放在服务器然后返回txt文本的内容到前台进行相应操作 前台html代码 <div id="coordinate_div">         ...

  9. java调用C/C++写的dll(转)

    源:java调用C/C++写的dll Java语言本身具有跨平台性,如果通过Java调用DLL的技术方便易用,使用Java开发前台界面可以更快速,也能带来跨平台性. Java调用C/C++写好的DLL ...

  10. leetcode--009 Linked List Cycle I

    aaarticlea/png;base64,iVBORw0KGgoAAAANSUhEUgAAAZgAAACACAIAAAC5q+hAAAAJ+UlEQVR4nO2dwbXrKBJAOyelRShEQw