0

我有以下代码来处理 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];
}

这可行,但我有一些问题,因为我通过拼凑我在网上找到的大量信息编写了这段代码。

  1. 第一个是第 5 行发生的事情以及它的正确名称。如果我没记错的话,这应该解压表达式并创建一个支撑初始化列表 [*]。这个列表的每个元素本身就是一个逗号分隔的列表,格式为 (exp,0)[**]。因此展开应该是 {(exp1,0),...(expn,0)}。我对吗?

  2. 第二个是关于评估顺序。[**] 应该只用于为 inizialier_list 的构造函数提供返回值而不用于其他目的(即使 index 本身可能是返回值?)。[*] 相反,给出了表达式求值的顺序。我在打印函数的示例中找到了它:

(void)std::initializer_list<int>{(print(others),0)...}; 

以便在参数上按顺序调用 print。这对于组合表达式也是如此吗?:

(void)std::initializer_list<int>{(print(compute(others)),0)...}; 

因此,如果“计算”依赖于状态,则会根据导致始终相同结果的参数顺序进行更新(独立于编译器 ecc ...)。在张量示例中,这指的是 ++i。

  1. 最后一个问题涉及到this,其中以对张量所做的方式使用 += 称为折叠表达式。我认为情况并非如此。我错了吗?
4

1 回答 1

3

1. 你的理解是正确的。

2.我不确定“组合表达式”等是什么意思,但是是的,大括号中的初始化程序总是从左到右进行评估。请参阅cppreference

  1. 在列表初始化中,给定初始值设定项子句的每个值计算和副作用都在每个值计算和与任何初始值项子句相关联的副作用之前进行排序,在括号括起来的逗号分隔的初始值设定项列表中。

3. 这不是折叠表达式。折叠表达式必须具有以下形式之一:

  • ( pack op ... )
  • ( ... op pack )
  • ( pack op ... op init )
  • ( init op ... op pack )

Where opis a binary operator(大多数是允许的),pack是一个包含至少一个未扩展参数包(Tsothers在您的情况下)init的表达式,只是一个普通表达式,并且...必须按字面意思存在。

可以使用折叠表达式简化您的函数。代替:

(void)std::initializer_list<int>{(index += _offset[++i]*static_cast<std::size_t>(others), 0)...};

你可以写:

((index += _offset[++i]*static_cast<std::size_t>(others)), ...);
于 2020-08-09T12:24:30.147 回答