名稱空間
變體
操作

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

來自 cppreference.com
< cpp‎ | ranges
 
 
範圍庫 (Ranges library)
範圍介面卡 (Range adaptors)
 
定義於標頭檔案 <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 用於指示一個 range 是否為 borrowed_range。主模板定義為 false

目錄

[編輯] 語義要求

Ustd::remove_reference_t<T> (如果 T 是右值引用型別),否則為 T。給定型別為 U 的變數 uT 建模 borrowed_range 僅當從 u 獲取的迭代器的有效性不與該變數的生命週期繫結。

[編輯] 特化

程式可以為建模 borrowed_range 的 cv-不限定程式定義型別特化 enable_borrowed_rangetrue,為不建模的型別特化為 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 不應返回,因為它會懸空
(class) [編輯]