如何在编译时进行条件变量初始化?

时间:2016-08-08 14:44:40

标签: c++ c++11

C ++ 11标准具有std::conditional<>模板,用于在编译器时通过某种布尔条件进行类型选择。 如何进行相同的操作但是为变量初始化选择init值?与type a = (exp) ? first_value : second_value;类似。

我使用我的模板:

template<bool B, typename T>
inline constexpr T&& conditional_initialize(T&& i1, T&& i2) {
    return B ? std::move(i1) : std::move(i2);
}

但它只能用于POD类型:int a = conditional_initialize<true>(1, 2);。 对于数组初始化,此模板编译时出错。错误的编译示例:int a[] = conditional_initialize<true>({1, 2}, {3,4,5});

  

错误讯息: no matching function for call to 'conditional_initialize(<brace-enclosed initializer list>, <brace-enclosed initializer list>)';

谁可以帮我模板?

1 个答案:

答案 0 :(得分:8)

template<class T, std::size_t N, std::size_t M, bool b>
std::array<T, b?N:M>
conditional_array( std::array<T, N>&& lhs, std::array<T, M>&& rhs ) {
  return std::move(std::get<b?0:1>( std::tie(lhs, rhs) ) );
}

这会给你:

auto a = conditional_array<int,2,3,true>({{1, 2}}, {{3,4,5}});

即将结束。

通常,{}构造不是表达式,它们不能通过任何机制通过另一个变量完美转发。

我们也可以得到:

auto a = cond_init<true>( make_array(1,2), make_array(3,4,5) );

有更多的工作。

template<bool Test, class A, class B>
std::conditional_t<Test,A,B>
cond_init(A a, B b) {
  return std::move( std::get<Test?0:1>( std::tie(a,b) ) );
}
template<class T0, class...Ts>
std::array< std::decay_t<T0>, sizeof...(Ts)+1 >
make_array( T0&& t0, Ts&&...ts ) {
  return {{std::forward<T0>(t0), std::forward<Ts>(ts)...}};
}

我没有制作这些constexpr,因为懒惰。