9

考虑以下代码:

struct A
{
    // No data members
    //...
};

template<typename T, size_t N>
struct B : A
{
    T data[N];
}

这就是你必须初始化 B 的方式:B<int, 3> b = { {}, {1, 2, 3} }; 我想避免基类不必要的空 {}。Jarod42 here提出了一个解决方案,但是,它不适用于元素默认初始化:B<int, 3> b = {1, 2, 3};很好但B<int, 3> b = {1};不是:b.data[1]并且b.data[2]没有默认初始化为0,并且会发生编译器错误。有什么方法(或者会有 c++20)从构造中“隐藏”基类?

4

4 回答 4

6

最简单的解决方案是添加可变参数构造函数:

struct A { };

template<typename T, std::size_t N>
struct B : A {
    template<class... Ts, typename = std::enable_if_t<
        (std::is_convertible_v<Ts, T> && ...)>>
    B(Ts&&... args) : data{std::forward<Ts>(args)...} {}

    T data[N];
};

void foo() {
    B<int, 3> b1 = {1, 2, 3};
    B<int, 3> b2 = {1};
}

如果您在{...}初始化列表中提供的元素少于N,则数组中的剩余元素data将被值初始化为 by T()

于 2019-12-27T10:23:18.143 回答
4

从 C++20 开始,您可以在聚合初始化中使用指定的初始化器。

B<int, 3> b = { .data {1} }; // initialize b.data with {1}, 
                             // b.data[0] is 1, b.data[1] and b.data[2] would be 0
于 2019-12-27T09:31:23.053 回答
4

仍然使用构造函数,您可能会执行以下操作:

template<typename T, size_t N>
struct B : A
{
public:
    constexpr B() : data{} {}

    template <typename ... Ts,
              std::enable_if_t<(sizeof...(Ts) != 0 && sizeof...(Ts) < N)
                               || !std::is_same_v<B, std::decay_t<T>>, int> = 0>
    constexpr B(T&& arg, Ts&&... args) : data{std::forward<T>(arg), std::forward<Ts>(args)...}
    {}

    T data[N];
};

演示

SFINAE 主要是为了避免创建伪拷贝构造函数B(B&)

你需要额外的私有标签来支持B<std::index_sequence<0, 1>, 42>;-)

于 2019-12-27T10:01:32.167 回答
2

我找到了另一个解决方案(我不知道如何)完美地工作并解决了我们在 Evg 的回答下讨论的问题

struct A {};

template<typename T, size_t N>
struct B_data
{
    T data[N];
};

template<typename T, size_t N>
struct B : B_data<T, N>, A
{
    // ...
};
于 2019-12-27T11:47:45.140 回答