我有以下代码来处理 n 维张量类(偏移量是 std::size_t 的 std::vector):
template <typename ...Ts>
double Tensor::at(int first, Ts... others) {
int i = 0;
std::size_t index = static_cast<std::size_t>(first)*_offset[0];
(void)std::initializer_list<int>{(index += _offset[++i]*static_cast<std::size_t>(others), 0)...};
return _data[index];
}
这可行,但我有一些问题,因为我通过拼凑我在网上找到的大量信息编写了这段代码。
第一个是第 5 行发生的事情以及它的正确名称。如果我没记错的话,这应该解压表达式并创建一个支撑初始化列表 [*]。这个列表的每个元素本身就是一个逗号分隔的列表,格式为 (exp,0)[**]。因此展开应该是 {(exp1,0),...(expn,0)}。我对吗?
第二个是关于评估顺序。[**] 应该只用于为 inizialier_list 的构造函数提供返回值而不用于其他目的(即使 index 本身可能是返回值?)。[*] 相反,给出了表达式求值的顺序。我在打印函数的示例中找到了它:
(void)std::initializer_list<int>{(print(others),0)...};
以便在参数上按顺序调用 print。这对于组合表达式也是如此吗?:
(void)std::initializer_list<int>{(print(compute(others)),0)...};
因此,如果“计算”依赖于状态,则会根据导致始终相同结果的参数顺序进行更新(独立于编译器 ecc ...)。在张量示例中,这指的是 ++i。
- 最后一个问题涉及到this,其中以对张量所做的方式使用 += 称为折叠表达式。我认为情况并非如此。我错了吗?