std::ranges::borrowed_range, std::ranges::enable_borrowed_range
来自cppreference.com
<tbody>
</tbody>
| 在标头 <ranges> 定义
|
||
template< class R > concept borrowed_range = ranges::range<R> && (std::is_lvalue_reference_v<R> || ranges::enable_borrowed_range<std::remove_cvref_t<R>>); |
(1) | (C++20 起) |
template< class R > constexpr bool enable_borrowed_range = false; |
(2) | (C++20 起) |
1) 概念
borrowed_range 定义对范围的要求,使得函数能按值接收它,并返回从它获得的迭代器,而无悬垂之虞。语义要求
令 U 当 T 为右值引用类型时是 std::remove_reference_t<T>,否则是 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 << ' '; // 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) |
占位类型,指示不应返回迭代器或子范围,因为它可能悬垂 (类) |