3

我想制作一个“接口”/混合类(模板),并确保没有人认为向这个类模板添加成员是个好主意,我想static_assert在这种情况下。不幸的是,std::is_empty不允许使用虚函数(因为多态类需要一些空间来存储它们的虚函数表指针或等效的实现细节)。

是否有std::is_empty允许虚拟功能但不允许数据成员的变体(或者它很容易编写)?

我希望这个工作(加上我现在想不出的所有黑暗角落案例):

#include <type_traits>

struct A {};
struct B : A {};
struct AA { virtual ~AA() = default; };
struct BB : AA {};

static_assert(std::is_empty_v<A>);
static_assert(std::is_empty_v<B>);

// These should also work with the replacement for is_empty:
static_assert(std::is_empty_v<AA>);
static_assert(std::is_empty_v<BB>);

int main()
{}

现场演示在这里

我猜它需要编译器魔术或非常特定于平台的黑客/检查,具体取决于虚拟函数的实现方式。

4

1 回答 1

3

我会这样做:

struct dummy_polymorphic
{
    virtual ~dummy_polymorphic() {}
};

template <typename T>
inline constexpr bool empty_maybe_polymorphic_v = std::is_empty_v<T> ||
    (std::is_polymorphic_v<T> && sizeof(T) <= sizeof(dummy_polymorphic));

这是我能想到的最好的方法,但它有一些局限性:

  • 我们假设多态类的开销总是相同的(通常是一个非常安全的假设,除非您的类由于多重继承而最终具有多个 vtable 指针)。我看不出有什么办法可以摆脱这个限制。

  • 如果T有重复的空碱基,它会中断:

    struct A {};
    struct B : A{};
    struct C : A, B {virtual ~C() {}};
    
    std::cout << sizeof(C) << '\n'; // 16, rather than 8
    
于 2019-11-15T14:10:04.890 回答