背景:我创建了以下类C
,其构造函数应该接受类型为B&
的N
个变量:
class A;
class B
{
A* getA();
};
template<size_t N>
class C
{
public:
template<typename... Args>
inline C(Args&... args) :
member{args.getA()...}
{}
private:
std::array<A*, N> member;
};
问题:我的问题是如何限制可变参数Args
的类型都是B
?
我的部分解决方案:我想定义一个谓词,像这样:
template <typename T, size_t N, typename... Args>
struct is_range_of :
std::true_type // if Args is N copies of T
std::false_type // otherwise
{};
并相应地重新定义我的构造函数:
template <typename... Args,
typename = typename std::enable_if<is_range_of_<B, N, Args...>::value>::type
>
inline C(Args&... args);
我在这篇文章中看到了一个可能的解决方案:https://dev59.com/HXA65IYBdhLWcg3wqAat#11414631,它定义了一个通用的check_all
谓词:
template <template<typename> class Trait, typename... Args>
struct check_all :
std::false_type
{};
template <template<typename> class Trait>
struct check_all<Trait> :
std::true_type
{};
template <template<typename> class Trait, typename T, typename... Args>
struct check_all<Trait, T, Args...> :
std::integral_constant<bool, Trait<T>::value && check_all<Trait, Args...>::value>
{};
所以,我可以写出类似这样的内容:
template <typename T, size_t N, typename... Args>
struct is_range_of :
std::integral_constant<bool,
sizeof...(Args) == N &&
check_all<Trait, Args...>::value
>
{};
问题1:我不知道如何定义Trait
,因为我需要以某种方式将std::is_same
与B
绑定为第一个参数。在我的情况下是否有使用通用的check_all
的方法,或者C++的当前语法不兼容?
问题2:我的构造函数也应该接受B
的派生类(通过对B
的引用),这会影响模板参数推断吗?我担心如果我使用像std::is_base_of
这样的谓词,我将为每组参数获得不同的构造函数实例化,这可能会增加编译后代码的大小...
编辑:例如,我有从B
继承的B1
和B2
,我在代码中调用C<2>(b1, b1)
和C<2>(b1, b2)
,它会创建两个实例(C<2>::C<B1, B1>
和C<2>::C<B1, B2>
)吗?我只想要C<2>::C<B, B>
的实例。
B
派生的,还是只是隐式转换为B
? - DeduplicatorB
,请看我的编辑。我需要一个模板泛化到一个定义构造函数C(B&b)
的类的N
个参数。 - SteakflyB
派生的参数既是比可转换更强的约束条件,也是更弱的约束条件。因为两者之间互不蕴含。 - Deduplicator