Можно ли предотвратить выделение стека объекта и разрешить его создание только с помощью 'new'?
Можно ли предотвратить выделение стека для объекта и разрешить его накапливание только с "новым" в куче?
6 ответов
Один из способов сделать это - сделать конструкторы приватными и разрешить конструирование только через статический метод, который возвращает указатель. Например:
class Foo
{
public:
~Foo();
static Foo* createFoo()
{
return new Foo();
}
private:
Foo();
Foo(const Foo&);
Foo& operator=(const Foo&);
};
В случае C++11
class Foo
{
public:
~Foo();
static Foo* createFoo()
{
return new Foo();
}
Foo(const Foo &) = delete; // if needed, put as private
Foo & operator=(const Foo &) = delete; // if needed, put as private
Foo(Foo &&) = delete; // if needed, put as private
Foo & operator=(Foo &&) = delete; // if needed, put as private
private:
Foo();
};
Вы могли бы сделать конструктор private
, а затем предоставить public
статический фабричный метод для создания объектов.
Следующее позволяет публичные конструкторы и остановит выделение стека, бросая во время выполнения. Заметка thread_local
является ключевым словом C++11
class NoStackBase {
static thread_local bool _heap;
protected:
NoStackBase() {
bool _stack = _heap;
_heap = false;
if (_stack)
throw std::logic_error("heap allocations only");
}
public:
void* operator new(size_t size) throw (std::bad_alloc) {
_heap = true;
return ::operator new(size);
}
void* operator new(size_t size, const std::nothrow_t& nothrow_value) throw () {
_heap = true;
return ::operator new(size, nothrow_value);
}
void* operator new(size_t size, void* ptr) throw () {
_heap = true;
return ::operator new(size, ptr);
}
void* operator new[](size_t size) throw (std::bad_alloc) {
_heap = true;
return ::operator new[](size);
}
void* operator new[](size_t size, const std::nothrow_t& nothrow_value) throw () {
_heap = true;
return ::operator new[](size, nothrow_value);
}
void* operator new[](size_t size, void* ptr) throw () {
_heap = true;
return ::operator new[](size, ptr);
}
};
bool thread_local NoStackBase::_heap = false;
Это должно быть возможно в C++20 с использованием уничтожающего оператора delete, см. P0722r3.
#include <new>
class C
{
private:
~C() = default;
public:
void operator delete(C *c, std::destroying_delete_t)
{
c->~C();
::operator delete(c);
}
};
Обратите внимание, что закрытый деструктор не позволяет использовать его для чего-либо другого, кроме динамического хранения. Но уничтожающий оператор delete позволяет уничтожить его с помощью выражения удаления (так как выражение удаления в этом случае неявно вызывает деструктор).
Вы можете создать заголовочный файл, который обеспечивает абстрактный интерфейс для объекта, и фабричные функции, которые возвращают указатели на объекты, созданные в куче.
// Header file
class IAbstract
{
virtual void AbstractMethod() = 0;
public:
virtual ~IAbstract();
};
IAbstract* CreateSubClassA();
IAbstract* CreateSubClassB();
// Source file
class SubClassA : public IAbstract
{
void AbstractMethod() {}
};
class SubClassB : public IAbstract
{
void AbstractMethod() {}
};
IAbstract* CreateSubClassA()
{
return new SubClassA;
}
IAbstract* CreateSubClassB()
{
return new SubClassB;
}