namespace std {
template<class I>
concept input_iterator =
input_or_output_iterator<I> &&
indirectly_readable<I> &&
requires { typename ITER_CONCEPT(I); } &&
derived_from<ITER_CONCEPT(I), input_iterator_tag>;
}
概要
input_iterator
は、イテレータ型I
が入力イテレータであることを表すコンセプトである。
input_iterator
となるイテレータは、operator*
による読み出しと前置/後置インクリメントによる進行が可能である。
ITER_CONCEPT
型I
について、std::iterator_traits<I>
がプライマリテンプレートの特殊化となる場合、ITER_TRAITS(I)
をI
とする。それ以外の場合(std::iterator_traits<I>
の特殊化が存在する場合)、ITER_TRAITS(I)
をstd::iterator_traits<I>
とする。
ITER_TRAITS(I)::itertor_concept
が有効で型名を示す場合、ITER_CONCEPT(I)
はITER_TRAITS(I)::itertor_concept
ITER_TRAITS(I)::itertor_category
が有効で型名を示す場合、ITER_CONCEPT(I)
はITER_TRAITS(I)::itertor_category
std::iterator_traits<I>
がプライマリテンプレートの特殊化となる場合、ITER_CONCEPT(I)
はrandom_access_iterator_tag
- 上記いずれにも当てはまらない場合、
ITER_CONCEPT(I)
は型名を示さない
例
#include <iostream>
#include <concepts>
#include <iterator>
#include <memory>
#include <vector>
template<std::input_iterator I>
void f(const char* name) {
std::cout << name << " is input_iterator" << std::endl;
}
template<typename I>
void f(const char* name) {
std::cout << name << " is not input_iterator" << std::endl;
}
struct sample_input_iterator {
// *thisの参照を返さなければならない(戻り値型は自身の参照型)
friend auto operator++(sample_input_iterator&) -> sample_input_iterator&;
// 戻り値型はvoidでも可
friend auto operator++(sample_input_iterator&, int) -> sample_input_iterator;
// const修飾されていなければならない
// indirectly_readableコンセプトにおいて、iter_reference_tの結果がconst有無両方で一致することが求められる
friend auto operator*(const sample_input_iterator&) -> int&;
// 少なくともこの3つは必須
using difference_type = int;
using value_type = int;
using iterator_concept = std::input_iterator_tag;
};
int main() {
f<int*>("int*");
f<const int*>("const int*");
f<std::vector<int>::iterator>("std::vector<int>::iterator");
f<std::istream_iterator<double>>("std::istream_iterator<double>");
f<sample_input_iterator>("sample_input_iterator");
std::cout << "\n";
f<int* const>("int* const");
f<std::ostream_iterator<double>>("std::ostream_iterator<double>");
}
出力
int* is input_iterator
const int* is input_iterator
std::vector<int>::iterator is input_iterator
std::istream_iterator<double> is input_iterator
sample_input_iterator is input_iterator
int* const is not input_iterator
std::ostream_iterator<double> is not input_iterator
バージョン
言語
- C++20
処理系
- Clang: ??
- GCC: 10.1 ✅
- Visual C++: 2019 Update 6 ✅