std::ranges::empty
来自cppreference.com
在标头 <ranges> 定义
|
||
在标头 <iterator> 定义
|
||
inline namespace /* 未指明 */ { inline constexpr auto empty = /* 未指明 */; |
(C++20 起) (定制点对象) |
|
调用签名 |
||
template< class T > requires /* 见下文 */ |
(C++20 起) | |
确定 t 是否拥有任何元素。
对 ranges::empty
的调用表达式等价于:
- bool(t.empty()),若该表达式合法。
- 否则为 (ranges::size(t) == 0),若该表达式合法。
- 否则为 bool(ranges::begin(t) == ranges::end(t)),若该表达式合法且 decltype(ranges::begin(t)) 实现 std::forward_iterator。
所有其他情况下,对 ranges::empty
的调用非良构,这能在 ranges::empty(t) 出现于模板实例化的立即语境中时导致代换失败。
定制点对象
名字 ranges::empty
代表一个定制点对象,它是某个字面 semiregular
类类型的 const 函数对象。为阐述目的,以 __empty_fn
表示它的类型的 cv 无限定版本。
__empty_fn
的所有实例均相等。在相同实参上调用 __empty_fn
类型的不同实例的效果是等价的,这与指代该实例的表达式是左值还是右值,以及是否为 const 限定无关(然而不要求 volatile 限定的实例是可调用的)。由此,可以自由地复制 ranges::empty
并且能彼此替代地使用它的副本。
给定类型集合 Args...
,如果 std::declval<Args>()... 满足上面对于 ranges::empty
的实参的要求,那么 __empty_fn
就实现
- std::invocable<__empty_fn, Args...>、
- std::invocable<const __empty_fn, Args...>、
- std::invocable<__empty_fn&, Args...> 和
- std::invocable<const __empty_fn&, Args...>。
否则,__empty_fn
的函数调用运算符不会参与重载决议。
示例
运行此代码
#include <iostream> #include <ranges> #include <vector> template<std::ranges::input_range R> void print(char id, R&& r) { if (std::ranges::empty(r)) { std::cout << '\t' << id << ") 为空\n"; return; } std::cout << '\t' << id << ") 包含元素:"; for (const auto& element : r) std::cout << ' ' << element; std::cout << '\n'; } int main() { { auto v = std::vector<int>{1, 2, 3}; std::cout << "(1) ranges::empty 使用 std::vector::empty:\n"; print('a', v); v.clear(); print('b', v); } { std::cout << "(2) ranges::empty 使用 ranges::size(initializer_list):\n"; auto il = {7, 8, 9}; print('a', il); print('b', std::initializer_list<int>{}); } { std::cout << "(2) ranges::empty 在原生数组上使用 ranges::size:\n"; int array[] = {4, 5, 6}; // 数组拥有已知边界 print('a', array); } { struct Scanty : private std::vector<int> { using std::vector<int>::begin; using std::vector<int>::end; using std::vector<int>::push_back; // 注:empty() 和 size() 都被隐藏 }; std::cout << "(3) 在没有 empty() 或 size() 的对象上调用 ranges::empty:\n"; Scanty y; print('a', y); y.push_back(42); print('b', y); } }
输出:
(1) ranges::empty 使用 std::vector::empty: a) 包含元素: 1 2 3 b) 为空 (2) ranges::empty 使用 ranges::size(initializer_list): a) 包含元素: 7 8 9 b) 为空 (2) ranges::empty 在原生数组上使用 ranges::size: a) 包含元素: 4 5 6 (3) 在没有 empty() 或 size() 的对象上调用 ranges::empty: a) 为空 b) 包含元素: 42
参阅
(C++17) |
检查容器是否为空 (函数模板) |