使用多个模板参数包应用 CTAD

tan*_*ngy 2 c++ templates variadic-templates c++17

遵循此处提供的一些解决方案如何在可变参数模板中包含多个参数包?,我希望将多个参数包应用于一个类并使用 CTAD 使该类更有用。

这是我想出的(在Coliru 上),但这给出了:

错误:类模板参数推导失败

这是我试过的代码:

// A template to hold a parameter pack
template < typename... >
struct Typelist {};

// Declaration of a template
template< typename TypeListOne 
        , typename TypeListTwo
        > 
struct Foo;     

// A template to hold a parameter pack
template <typename... Args1, typename... Args2>
struct Foo< Typelist < Args1... >
                 , Typelist < Args2... >
                 >{
    template <typename... Args>
    struct Bar1{
        Bar1(Args... myArgs) {
            _t = std::make_tuple(myArgs...);
        }
        std::tuple<Args...> _t;
    };

    template <typename... Args>
    struct Bar2{
        Bar2(Args... myArgs) {
            _t = std::make_tuple(myArgs...);
        }
        std::tuple<Args...> _t;
    };

    Bar1<Args1...> _b1;
    Bar2<Args2...> _b2;

    Foo(Bar1<Args1...>& b1, Bar2<Args2...>& b2) {
        _b1 = b1;
        _b2 = b2;
    }
};

int main()
{

    Foo{Foo::Bar1(1, 2.0, 3), Foo::Bar2(100, 23.4, 45)};
    return 0;
}
Run Code Online (Sandbox Code Playgroud)

wal*_*nut 6

首先,CTAD 在使用范围解析操作符的时候没有做,所以Foo::永远不能使用。您必须为此Foo明确指定模板参数。

我建议你只需移动Bar1Bar2外部Foo偏特到的命名空间范围和使用

Foo{Bar1(1, 2.0, 3), Bar2(100, 23.4, 45)};
Run Code Online (Sandbox Code Playgroud)

反而。


然后你会得到一个错误,推断Foo失败。这是因为隐式推导指南只考虑模板中的构造函数。但是您要使用的构造函数位于部分特化中,而不是主模板中。

因此,您需要自己添加适当的扣除指南,例如:

template <typename... Args1, typename... Args2>
Foo(Bar1<Args1...>, Bar2<Args2...>) -> Foo<Typelist<Args1...>, Typelist<Args2...>>;
Run Code Online (Sandbox Code Playgroud)

然后你会得到一个错误,说没有构造函数是可行的。这是因为您将Bar1<Args1...>& b1andBar2<Args2...>& b2作为非const左值引用,但您为它们提供了纯右值。非const左值引用不能绑定到右值,所以你会得到一个错误。通过值或const左值引用获取参数。


最后你会得到一个错误,_b1并且_b2没有默认构造函数,这是真的。它们是必需的,因为您正在默认初始化_b1并且_b2在您的构造函数中。您稍后才为它们分配值。

因此,要么添加默认构造函数Bar1Bar2要么更好地使用初始化而不是赋值:

Foo(const Bar1<Args1...>& b1, const Bar2<Args2...>& b2) : _b1(b1), _b2(b2) { }
Run Code Online (Sandbox Code Playgroud)

在所有这些步骤之后,代码应该可以编译了。我不确定你的目标到底是什么,所以不完全确定这是否会做你想要的。