3

std::vector几乎所有其他容器都有一种非常方便的边界检查方法:at(). std::span显然没有。

4

2 回答 2

2

相当笨重,但像这样:

  1. 使用位置
template<class Container>
auto& at(Container&& c, std::size_t pos){
    if(pos >= c.size())
        throw std::out_of_range("out of bounds");
    return c[pos];
}
  1. 使用迭代器:
template<class Iterator, class Container>
auto& at(Container&& c, Iterator&& it){
    if(std::distance(c.begin(), it) >= c.size())
        throw std::out_of_range("out of bounds");
    return *it;
}
于 2020-08-02T12:12:58.247 回答
2

将 span 引入标准库的论文说:

范围检查和边界安全

对跨度封装的数据的所有访问在概念上都经过范围检查,以确保它们保持在跨度的范围内。由于在运行时未能满足 span 的 boundssafety 约束而实际发生的情况是未定义的行为。

也就是说,操作具有狭窄的合同,为实施者提供了自由。

如果您的标准库不允许您将行为控制到适当的粒度。gsl-lite提供了可配置的合同违规行为的替换。Microsoft GSL以前是可配置的,但现在总是在违反合同时终止,在此处讨论(这实际上可能是您想要的)。

于 2020-08-02T12:19:36.897 回答