std::ranges::borrowed_range, std::ranges::enable_borrowed_range
来自 cppreference.cn
定义于头文件 <ranges> |
||
template< class R > concept borrowed_range = |
(1) | (C++20 起) |
template< class R > constexpr bool enable_borrowed_range = false; |
(2) | (C++20 起) |
1) `borrowed_range` 概念定义了范围的要求,使得函数可以按值接受它并返回从中获取的迭代器,而不会有悬垂的危险。
2) `enable_borrowed_range` 变量模板用于指示一个范围是否是 `borrowed_range`。主模板定义为
false
。目录 |
[编辑] 语义要求
令 U 为 std::remove_reference_t<T>
(如果 T 是右值引用类型),否则为 T。给定类型为 U 的变量 u,仅当从 u 获取的迭代器的有效性不依赖于该变量的生命周期时,T 才建模 borrowed_range
。
[编辑] 特化
对于建模 borrowed_range
的 cv 限定的程序定义类型,程序可以将 enable_borrowed_range
特化为 true
,对于不建模 borrowed_range
的类型,则特化为 false
。这些特化应可在常量表达式中使用,并且类型为 const bool
。
[编辑] 标准库中无条件借用的范围
对于以下标准模板的所有特化,enable_borrowed_range
的特化定义为 true
- std::basic_string_view
- std::span
- std::ranges::subrange
- std::ranges::ref_view
- std::ranges::empty_view
- std::ranges::iota_view
[编辑] 标准库中有条件借用的范围
对于以下标准范围适配器,enable_borrowed_range
的特化被定义为 true
当且仅当 std::ranges::enable_borrowed_range<V>
为 true
,其中 V 是底层视图类型。
(C++23 起) |
- std::ranges::common_view
- std::ranges::drop_view
- std::ranges::drop_while_view
- std::ranges::elements_view
(C++23 起) |
(C++23 起) |
(C++26 起) |
- ↑ 底层视图 V 也必须满足
forward_range
。
对于以下标准范围适配器, |
(C++23 起) |
[编辑] 示例
演示了程序定义类型的 enable_borrowed_range
特化。这些特化防止潜在的悬垂结果。
运行此代码
#include <algorithm> #include <array> #include <cstddef> #include <iostream> #include <ranges> #include <span> #include <type_traits> template<typename T, std::size_t N> struct MyRange : std::array<T, N> {}; template<typename T, std::size_t N> constexpr bool std::ranges::enable_borrowed_range<MyRange<T, N>> = false; template<typename T, std::size_t N> struct MyBorrowedRange : std::span<T, N> {}; template<typename T, std::size_t N> constexpr bool std::ranges::enable_borrowed_range<MyBorrowedRange<T, N>> = true; int main() { static_assert(std::ranges::range<MyRange<int, 8>>); static_assert(std::ranges::borrowed_range<MyRange<int, 8>> == false); static_assert(std::ranges::range<MyBorrowedRange<int, 8>>); static_assert(std::ranges::borrowed_range<MyBorrowedRange<int, 8>> == true); auto getMyRangeByValue = []{ return MyRange<int, 4>{{1, 2, 42, 3}}; }; auto dangling_iter = std::ranges::max_element(getMyRangeByValue()); static_assert(std::is_same_v<std::ranges::dangling, decltype(dangling_iter)>); // *dangling_iter; // compilation error (i.e. dangling protection works.) auto my = MyRange<int, 4>{{1, 2, 42, 3}}; auto valid_iter = std::ranges::max_element(my); std::cout << *valid_iter << ' '; // OK: 42 auto getMyBorrowedRangeByValue = [] { static int sa[4]{1, 2, 42, 3}; return MyBorrowedRange<int, std::size(sa)>{sa}; }; auto valid_iter2 = std::ranges::max_element(getMyBorrowedRangeByValue()); std::cout << *valid_iter2 << '\n'; // OK: 42 }
输出
42 42
[编辑] 参见
(C++20) |
一种占位符类型,指示不应返回迭代器或 subrange ,因为它会是悬垂的(类) |