为什么我不能用大括号括起来的初始化列表构造 gsl::span
Why can't I construct a gsl::span with a brace-enclosed initializer list
根据C++ Core Guidelines, I should use a gsl::span to pass a half-open sequence。
我认为这意味着与其编写如下函数:
void func(const std::vector<int>& data) {
for (auto v : data) std::cout << v << " ";
}
我更喜欢:
void func(gsl::span<const int> data) {
for (auto v : data) std::cout << v << " ";
}
它的优点是它不会假定调用者的数据在 vector
中,也不会强制他们构建一个临时的 vector
。例如,他们可以传递 std::array
。
但是一个常见的用例是传递一个大括号括起来的初始化列表:
func({0,1,2,3})
这适用于采用 std::vector
的函数,但对于采用 gsl::span
的函数我收到错误消息:
error C2664: 'void func(gsl::span)' : cannot convert
argument 1 from 'initializer-list' to 'gsl::span'
它看起来像 gsl::span
has a templated constructor 旨在容纳任何容器。
这只是 Microsoft GSL 实施中缺少的东西,还是有充分的理由阻止这种做法?
当您调用矢量版本时,初始化列表用于创建一个临时的 std::vector
,然后通过 const 引用将其传递给函数。这是可能的,因为 std::vector
有一个构造函数,它接受一个 std::initializer_list<T>
作为参数。
但是, gsl::span
没有这样的构造函数,并且 {0,1,2,3}
没有类型,它也不能被您提到的模板化构造函数接受(除此之外, std::initializer_list<T>
无论如何都不能满足容器的概念。
一个(丑陋的)解决方法当然是显式创建一个临时数组:
func(std::array<int,4>{ 0,1,2,3 });
我没有看到一个特别的原因,为什么 gsl::span
不应该有一个采用 std::initializer_list
的构造函数,但请记住,这个库仍然很新并且正在积极开发中.所以也许这是他们忽略的事情,没有时间实施,不确定如何正确地做或者确实有一些细节,这会使该结构变得危险。最好直接在 github.
上询问开发人员
编辑:
正如@Nicol Bolas 在他的评论中解释的那样,这是 by design 因为像 {0,1,2,3}
这样的初始化列表(以及其中的元素)是一个临时对象,而 gsl::span
本身不是一个容器对(它不拥有元素的所有权),他们认为不小心创建一个包含对这些临时元素的悬空引用的 gsl::span
太容易了。
所以,虽然这没问题:
func({ 0,1,2,3 });
因为初始化列表的生命周期在函数完成后结束,像这样的事情会创建一个悬空引用:
gsl::span<const int> data{ 0,1,2,3 };
func(data);
跨度是非拥有的。不拥有存储空间。它是指针运算的替代品,而不是存储 class.
你需要把你的数据放在一个存储中class,然后如果你想用指针算法做一些聪明的事情,你可以用跨度做一些聪明的事情。
您不能使用初始化列表初始化跨度,因为没有地方可以放置数据。
这现在适用于 absl::Span
。下面的例子是从https://abseil.io/tips/93复制过来的:
void TakesSpan(absl::Span<const int> ints);
void PassALiteral() {
// Span does not need a temporary allocation and copy, so it is faster.
TakesSpan({1, 2, 3});
}
根据C++ Core Guidelines, I should use a gsl::span to pass a half-open sequence。
我认为这意味着与其编写如下函数:
void func(const std::vector<int>& data) {
for (auto v : data) std::cout << v << " ";
}
我更喜欢:
void func(gsl::span<const int> data) {
for (auto v : data) std::cout << v << " ";
}
它的优点是它不会假定调用者的数据在 vector
中,也不会强制他们构建一个临时的 vector
。例如,他们可以传递 std::array
。
但是一个常见的用例是传递一个大括号括起来的初始化列表:
func({0,1,2,3})
这适用于采用 std::vector
的函数,但对于采用 gsl::span
的函数我收到错误消息:
error C2664: 'void func(gsl::span)' : cannot convert argument 1 from 'initializer-list' to 'gsl::span'
它看起来像 gsl::span
has a templated constructor 旨在容纳任何容器。
这只是 Microsoft GSL 实施中缺少的东西,还是有充分的理由阻止这种做法?
当您调用矢量版本时,初始化列表用于创建一个临时的 std::vector
,然后通过 const 引用将其传递给函数。这是可能的,因为 std::vector
有一个构造函数,它接受一个 std::initializer_list<T>
作为参数。
但是, gsl::span
没有这样的构造函数,并且 {0,1,2,3}
没有类型,它也不能被您提到的模板化构造函数接受(除此之外, std::initializer_list<T>
无论如何都不能满足容器的概念。
一个(丑陋的)解决方法当然是显式创建一个临时数组:
func(std::array<int,4>{ 0,1,2,3 });
我没有看到一个特别的原因,为什么 gsl::span
不应该有一个采用 std::initializer_list
的构造函数,但请记住,这个库仍然很新并且正在积极开发中.所以也许这是他们忽略的事情,没有时间实施,不确定如何正确地做或者确实有一些细节,这会使该结构变得危险。最好直接在 github.
编辑:
正如@Nicol Bolas 在他的评论中解释的那样,这是 by design 因为像 {0,1,2,3}
这样的初始化列表(以及其中的元素)是一个临时对象,而 gsl::span
本身不是一个容器对(它不拥有元素的所有权),他们认为不小心创建一个包含对这些临时元素的悬空引用的 gsl::span
太容易了。
所以,虽然这没问题:
func({ 0,1,2,3 });
因为初始化列表的生命周期在函数完成后结束,像这样的事情会创建一个悬空引用:
gsl::span<const int> data{ 0,1,2,3 };
func(data);
跨度是非拥有的。不拥有存储空间。它是指针运算的替代品,而不是存储 class.
你需要把你的数据放在一个存储中class,然后如果你想用指针算法做一些聪明的事情,你可以用跨度做一些聪明的事情。
您不能使用初始化列表初始化跨度,因为没有地方可以放置数据。
这现在适用于 absl::Span
。下面的例子是从https://abseil.io/tips/93复制过来的:
void TakesSpan(absl::Span<const int> ints);
void PassALiteral() {
// Span does not need a temporary allocation and copy, so it is faster.
TakesSpan({1, 2, 3});
}