如何在编译时使用std :: make_tuple?

时间:2019-02-15 03:49:30

标签: c++14 std c++17 constexpr

由于使用std :: make_tuple,返回std::array<std:tuple<uint32_t, uint32_t, uint32_t>, size_t>的Constexpr函数在编译时不起作用。有什么办法可以克服这个问题?

当我尝试删除constexpr规范时。它可以正常工作。但是,我们项目的目标是在编译时提供这种功能评估。

我遇到以下错误:

在通话部分:

error: call to non-constexpr function ‘std::tuple<_Elements>& std::tuple<_Elements>::operator=(std::tuple<_Elements>&&) [with _Elements = {unsigned int, unsigned int, unsigned int}]’

在功能部分:

error: ‘constexpr std::array<std::tuple<unsigned int, unsigned int, unsigned int>, SIZE> GenArrayTuple() [with long unsigned int SIZE = 128]’ called in a constant expression

代码在下面。

template<std::size_t SIZE>
constexpr std::array<std::tuple<uint32_t, uint32_t, uint32_t>, SIZE> 
GenArrayTuple() {
  std::array<std::tuple<uint32_t, uint32_t, uint32_t>, SIZE> array;
  for (uint32_t i = 0; i < SIZE; ++i) {
    // FIXME constexpr
    arr[2*i] = std::make_tuple(i, i * 2, i * 3 + 1);
  }
  return array;
}

constexpr uint32_t n = 128; 
constexpr auto array_tuple = GenArrayTuple<n>();

1 个答案:

答案 0 :(得分:1)

在C ++ 14或更高版本的常量表达式中使用std::make_tuple实际上没有问题,因为C ++ 14将其更改为constexpr。因此,这是一个有效的常量表达式,只要用于初始化元组元素的任何类构造函数都将其评估为有效的常量表达式(并且当您的元素类型都是标量为std::uint32_t时,就没有此类构造函数)。

但是请更好地查看错误消息。它抱怨的功能是(删除一些细节)tuple& tuple::operator=(tuple&&)。事实证明,在当前C ++版本中,std::tuple的{​​{3}}没有标记为constexpr,这意味着tuple对象的任何赋值都不是有效的常量表达式。 (cppreference.com指出,它们将在C ++ 20中标记为constexpr;通常反映了相应C ++工作组已经接受的提案的更改。)

因此,要解决此问题,您需要一次全部初始化array,而不是在循环中分配其元素。可能最简单的方法是借助assignment operators

#include <tuple>
#include <array>
#include <cstdint>
#include <utility>

template <std::uint32_t ... I>
constexpr std::array<std::tuple<std::uint32_t, std::uint32_t, std::uint32_t>,
                     sizeof...(I)>
GenArrayTuple_helper(std::integer_sequence<std::uint32_t, I...>) {
    return { std::make_tuple(I, I * 2, I * 3 + 1) ... };
}

template <std::size_t SIZE>
constexpr std::array<std::tuple<std::uint32_t, std::uint32_t, std::uint32_t>,
                     SIZE> 
GenArrayTuple() {
    return GenArrayTuple_helper(std::make_integer_sequence<std::uint32_t, SIZE>{});
}