10

我有类 Data 可以保存指向对象的指针。我希望稍后能够手动调用它的析构函数,为此我需要将其地址存储在一个变量中,但似乎禁止获取构造函数/析构函数的地址。有没有办法解决 ?

struct Data {

  union {
    long i;
    float f;
    void* data_ptr;
  } _data;

  std::type_index _typeIndex;
  void (*_destructor_ptr)();

  template<typename T>
  void Init() {
  if constexpr (std::is_integral<T>::value) {
    //
  }
  else if constexpr (std::is_floating_point<T>::value) {
    //
  }
  else {
    _data.data_ptr = new T;
    _typeIndex = std::type_index(typeid(T));
    _destructor_ptr = &T::~T; // << -- can't do this
  }
}
4

3 回答 3

9

存储一个 lambda,适当转换:

void (*_destructor_ptr)(void *v);

// ...

_destructor_ptr = [](void* v) { delete static_cast<T*>(v); };

请注意,您必须通过_data.data_ptrfor v。如果您打算存储普通函数指针,则 lambda 可能不会捕获或隐式引用_data.data_ptr.

于 2018-10-10T09:56:30.620 回答
7

如果您的编译器不支持 lambda,还有这个解决方案:

template<typename T>
struct DestructorHelper {
  static void Destroy(void * v) {
    delete static_cast<T*>(v);
  }
};

并将其用作:

_destructor_ptr = &DestructorHelper<T>::Destroy;
于 2018-10-10T12:51:16.887 回答
1

我还将为智能指针添加一个解决方案:

template <class T>
struct DataPtrDeleter
{
    void operator()(void * p) { delete (T*)p; }
}

std::shared_ptr<void*> data_ptr(new T, DataPtrDeleter<T>());
//or
std::unique_ptr<void*, DataPtrDeleter<T>> data_ptr(new T, DataPtrDeleter<T>());
于 2018-10-10T14:14:34.653 回答