如何在编译时使用std :: make_tuple
How can I use std::make_tuple at compile time?
返回std::array<std:tuple<uint32_t, uint32_t, uint32_t>, size_t>
的constexpr函数由于使用std :: make_tuple而在编译时间内不起作用。有什么方法可以克服这一点?
当我尝试删除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>();
在C 14或更高版本中使用std::make_tuple
实际上没有问题,因为C 14将其更改为constexpr
。因此,只要任何用于初始化元组元素的类构造函数评估为有效的常数表达式(当您的元素类型都是 std::uint32_t
等标量时,就没有这样的构造函数。
,但可以更好地查看错误消息。它抱怨的功能是(删除一些细节)tuple& tuple::operator=(tuple&&)
。事实证明,std::tuple
的分配运算符在当前C 版本中未标记为constexpr
,这意味着tuple
对象的任何分配都不是有效的常数表达式。(cppreference.com指出,它们将在C 20中标记为constexpr
;这通常反映了相应C 工作组已经接受的建议的变化。)
为了解决此问题,您需要一次初始化array
,而不是在循环中分配其元素。可能最简单的方法是借助std::make_integer_sequence
:
#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>{});
}
相关文章:
- 使用std::multimap迭代器创建std::list
- C++中std::resize(n)和std::shrink_to_fit之间的区别
- 来自 std::list 的迭代器 .end() 按预期返回"0xcdcdcdcdcdcdcdcd"但 .begin()
- C++17复制构造函数,在std::unordereded_map上进行深度复制
- 如何导出包含具有"std::unique_ptr"值的"std::map"属性的
- 从持续时间构造std::chrono::system_clock::time_point
- std::具有相同基类的类的变体
- std::向量与传递值的动态数组
- 使用std::vector的OpenCL矩阵乘法
- std::map<struct,struct>::find 找不到匹配项,但是如果我循环通过 begin() 到 end(),我在那里看到匹配项
- std::condition_variable::wait()如何评估给定的谓词
- 如何获取std::result_of函数的返回类型
- std::原子加载和存储都需要吗
- 将对象移动到std::shared_ptr
- 如何摆脱导入的 make 项目中的 Eclipse 索引器"Type std::... could not be resolved"错误
- Make zmqpp::socket::connect a std::future
- 为什么是 std::make_unique 而不是 std::unique_ptr::make?
- Travis ci make -std=c++14无法识别
- const decltype(*std::begin(container))& val dont make val const?
- make and std::logic_error