命名空间
变体
操作

std::ranges::borrowed_range, std::ranges::enable_borrowed_range

来自 cppreference.cn
< cpp‎ | ranges
 
 
范围库
范围适配器
 
定义于头文件 <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` 概念定义了范围的要求,使得函数可以按值接受它并返回从中获取的迭代器,而不会有悬垂的危险。
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

[编辑] 标准库中有条件借用的范围

对于以下标准范围适配器,enable_borrowed_range 的特化被定义为 true 当且仅当 std::ranges::enable_borrowed_range<V>true,其中 V 是底层视图类型。

(C++23 起)
(C++23 起)
(C++23 起)
(C++26 起)
  1. 底层视图 V 也必须满足 forward_range

对于以下标准范围适配器,enable_borrowed_range 的特化被定义为 true 当且仅当 (std::ranges::enable_borrowed_range<Vs> && ...)true,其中 Vs... 是它适配的所有视图类型。

(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

[编辑] 参见

一种占位符类型,指示不应返回迭代器或 subrange,因为它会是悬垂的
(类) [编辑]