您的位置:首页 > 其它

如何限制一个类对象只在堆上或栈上建立

2018-02-25 18:45 176 查看

类对象的创建在栈上:

静态分配类对象:编译器直接调用类的构造函数,挪动栈顶指针为对象分配适当的内存空间。

所以要使类对象建立在栈上就必须采用静态分配类对象的方法(直接调用类的构造函数),禁用new运算符 (new首先要调用operator new为对象在堆上分配合适的内存的内存,再调用类的构造函数进行初始化) 将operator new函数声明成私有,这时operator delete函数也应声明成private。

class Base
{
private:
void* operator new(size_t t){}
void operator delete(void* ptr){}
public:
Base(){}
~Base(){}
};
int main()
{
Base B;
};


类对象的创建在堆上:

动态分配类对象:利用new运算符调用operator new函数在堆空间寻找适合的内存空间分配给对象,再调用构造函数对内存进行初始化。

要使类对象建立在堆上,需使用new运算符进行对象的创建,不能直接调用构造函数。这时脑袋中可能会有一种想法——将构造函数设置为私有,但是new运算符在创建对象的时候也是要调用构造函数的,所以不能解决问题。

当对象在栈上建立的时候,编译器为对象分配空间,调用构造函数进行初始化,当对象使用过后编译器需调用析构函数清理内存;如果不能编译器不能调用到析构函数,就不能将对象申请的内存空间进行释放。从中可知:编译器在栈上建立对象时需考虑析构函数是不是可以在对象使用完之后调用,如果不能调用就不能在栈上建立类对象。

将析构函数设置为私有就只能在堆上建立类对象。

这里又出现了问题:当该类的析构函数(virtual)设置为私有的时候遇上该类做基类,派生类就不能调用基类的析构函数释放基类对象内存。这时可以将析构函数设置为protected,派生类可以调用,问题解决。

class Base
{
public:
Base(){}
void Destory(){delete this;}
// 直接使用delete会报错,因为它会调用析构函数,
// 需用Destory函数将delete封装在类内。
//这里也可以将析构函数设置为保护即可解决
private:
~Base(){}
};
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签:  堆栈建立类对象