今天遇到一个很有意思的面试题:
面试官:“怎么让类只能在堆上生成实例?”
我:“把析构函数设为私有”
原因如下:
编译器在为类对象分配栈空间时,会先检查类的析构函数的访问性,其实不光是析构函数,只要是非静态的函数,编译器都会进行检查。如果类的析构函数是私有的,则编译器不会在栈空间上为类对象分配内存。
当然,之后必须提供一个public的函数删除对象。
class test
{
private:
~test(){ cout << "test destroy" << endl; }
public:
void destroy()
{
delete this;
}
};
int main()
{
//test p;//编译器报错test::~test()不可访问
test *p = new test;
p->destroy();
}
当然啦,如果需要只在栈上创建只需要重载new即可
class test
{
private:
void* operator new(size_t t){}
void operator delete(void* ptr){}
public:
~test()
{
cout << "test destroy" << endl;
}
};
int main()
{
test *A = new test; //编译器报错
//函数test::operator new 不可访问
test A;//对
}
可是我这个回答面试官貌似并不满意的样子???也不给我学(fan)习(wen)的机会,sad。
2020.10.8 更新:
破案了。一旦将构造函数设为私有尽管只能在堆上生成,但是忽略了继承的问题。这时候应该当想到protected,即类外无法访问但子类可以访问。