如何扩展integer_sequence?

时间:2019-06-14 18:10:42

标签: c++ templates sequence expansion variable-templates

我有一个看起来像这样的函数:

template <typename T, std::size_t... I>
std::ostream& vector_insert(std::ostream& lhs, const char* delim, const T& rhs, std::index_sequence<I...>) {
    std::ostream_iterator<float> it(lhs, delim);

    ((*it++ = at(rhs, I)), ...);
    return lhs;
}

这是我最后的尝试,但我仍然无法扩展integer_sequence,希望有人可以告诉我如何编写可以有效扩展到以下内容的行:

*it++ = at(rhs, 0U), *it++ = at(rhs, 1U), *it++ = at(rhs, 2U)

我尝试过的其他事情是:

  1. *it++ = at(rhs, I...)
  2. *it++ = at(rhs, I)...
  3. (*it++ = at(rhs, I))...

所有人都给我错误:

  

错误C3520:I:必须在这种情况下扩展参数包

我如何扩展这个东西?

编辑:

@AndyG has pointed out,这似乎是的错误。

1 个答案:

答案 0 :(得分:2)

这似乎是Visual C ++的编译器错误。除了简化扩展参数包的表达式外,我不知道有任何简单的解决方法。转换为递归方法似乎可以可靠地解决此问题。例如:

#include <array>
#include <iostream>
#include <iterator>

template <typename T>
const auto& at(const T& v, size_t i) { return v[i]; }

// End of recursion
template<class T>
void vector_insert_impl(std::ostream_iterator<int> &, const char*, const T&)
{}

// Recursion case
template<class T, std::size_t N, std::size_t... I>
void vector_insert_impl(std::ostream_iterator<int> & iter, const char* delim, const T&rhs)
{
    *iter++ = at(rhs, N);

    // Continue the recursion
    vector_insert_impl<T, I...>(iter, delim, rhs);
}

template <typename T, std::size_t... I>
std::ostream& vector_insert(std::ostream& lhs, const char* delim, const T& rhs, std::index_sequence<I...>) 
{
    std::ostream_iterator<int> it(lhs, delim);

    // Call the recursive implementation instead
    vector_insert_impl<T, I...>(it, delim, rhs);

    return lhs;
}

int main() {
    std::array<int, 5> v = { 1, 2, 3, 4, 5 };
    vector_insert(std::cout, " ", v, std::make_index_sequence<v.size()>());
}

在这里,仅在提供VC ++不会遇到麻烦的模板参数的上下文中扩展参数包I