std::ranges:: borrowed_range, std::ranges:: enable_borrowed_range
|
定义于头文件
<ranges>
|
||
|
template
<
class
R
>
concept borrowed_range
=
|
(1) | (C++20 起) |
|
template
<
class
R
>
constexpr bool enable_borrowed_range = false ; |
(2) | (C++20 起) |
borrowed_range
定义了这样的范围要求:函数可以按值接收该范围,并返回从其获取的迭代器,而不会产生悬空危险。
目录 |
语义要求
若
T
为右值引用类型,则令
U
为
std::
remove_reference_t
<
T
>
,否则
U
为
T
。给定类型为
U
的变量
u
,仅当从
u
获取的迭代器有效性不依赖于该变量生命周期时,
T
才满足
borrowed_range
概念。
特化
程序可以为满足
borrowed_range
概念的
程序定义类型
(cv 未限定)将
enable_borrowed_range
特化为
true
,并为不满足的类型特化为
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; // 编译错误(即悬空保护生效) auto my = MyRange<int, 4>{{1, 2, 42, 3}}; auto valid_iter = std::ranges::max_element(my); std::cout << *valid_iter << ' '; // 正常: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'; // 正常:42 }
输出:
42 42
参见
|
(C++20)
|
用于指示不应返回迭代器或
subrange
的占位符类型,因为返回会导致悬空引用
(类) |