我有一些辅助代码使用编译时索引执行矢量重新洗牌。最重要的是生成的代码尽可能高效。我依赖于带有折叠表达式的参数包,我想知道编写此类代码的最佳实践是什么。
一个实际的例子:假设有一个函数insert
将容器的元素插入y
到容器x
中的位置Ii
,这些位置是编译时常量。这个函数的基本签名是这样的:
template<size_t... Ii, size_t Xsize, size_t Size>
constexpr container<Xsize> insert(container<Xsize> x, container<Ysize> y);
它的调用是这样的:insert<0, 2>(x, y)
. 我看到了实现这一点的两种明显的可能性。
第一:使用辅助索引变量迭代y
:
template<size_t... Ii, size_t Xsize, size_t Size>
constexpr container<Xsize> insert(container<Xsize> x, container<Ysize> y) {
int i = 0;
((x[Ii] = y[i++]), ...);
return x;
}
我对这个解决方案的问题是变量i
:我必须依靠编译器来优化它。
第二种解决方案避免了任何运行时依赖,但它需要一个辅助函数,使得整个实现相当难看:
template<size_t... Ii, size_t... Yi, size_t Xsize, size_t Size>
constexpr container<Xsize> insert_(container<Xsize> x, container<Ysize> y, std::index_sequence<Yi...>) {
((x[Ii] = y[Yi]), ...);
return x;
}
template<size_t... Ii, size_t Xsize, size_t Size>
constexpr container<Xsize> insert(container<Xsize> x, container<Ysize> y) {
return insert_<Ii...>(x,y, std::make_index_sequence<sizeof...(Ii)> {});
}
有没有办法避免运行时变量和辅助函数?