std::ranges::borrowed_range, std::ranges::enable_borrowed_range
來自 cppreference.com
定義於標頭檔案 <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
定義了這樣一種範圍的要求:函式可以透過值傳遞它,並返回從中獲取的迭代器而沒有懸空危險。目錄 |
[編輯] 語義要求
設 U
是 std::remove_reference_t<T> (如果 T
是右值引用型別),否則為 T
。給定型別為 U
的變數 u,T
建模 borrowed_range
僅當從 u 獲取的迭代器的有效性不與該變數的生命週期繫結。
[編輯] 特化
程式可以為建模 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; // 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 不應返回,因為它會懸空(class) |