There are a number of different ways, that make a type/class usable in a ranged for loop. An overview is for example given on cppreference:
range-expression
is evaluated to determine the sequence or range to iterate. Each element of the sequence, in turn, is dereferenced and is used to initialize the variable with the type and name given in range-declaration.
begin_expr
andend_expr
are defined as follows:
- If
range-expression
is an expression of array type, thenbegin_expr
is__range
andend_expr
is (__range
+__bound
), where__bound
is the number of elements in the array (if the array has unknown size or is of an incomplete type, the program is ill-formed)- If
range-expression
is an expression of a class typeC
that has both a member namedbegin
and a member namedend
(regardless of the type or accessibility of such member), thenbegin_expr
is__range.begin()
andend_expr
is__range.end()
- Otherwise,
begin_expr
isbegin(__range)
andend_expr
isend(__range)
, which are found via argument-dependent lookup (non-ADL lookup is not performed).
If I want to use a ranged for loop say in a function template, I want to constrain the type to be usable in a ranged for loop, to trigger a compiler error with a nice "constraint not satisfied" message. Consider the following example:
template<typename T>
requires requires (T arg) {
// what to write here ??
}
void f(T arg) {
for ( auto e : arg ) {
}
}
Obviously for a generic function I want to support all of the above listed ways, that a type can use to make itself ranged for compatible.
This brings me to my questions:
begin
and end
regardless of the type or accessibility of such member (second bullet in the qouted
list). A requires clause that tests for example for the existence of a begin()
member fails if begin()
is private, but the ranged for loop would be able to use
the type regardless of that.Note I am aware of the following two questions:
but neither of them really answers my question.
It seems like what you need is std::ranges::range
which requires the expressions ranges::begin(t)
and ranges::end(t)
to be well-formed.
Where ranges::begin
is defined in [range.access.begin]:
The name
ranges::begin
denotes a customization point object. Given a subexpressionE
with typeT
, let t be an lvalue that denotes the reified object forE
. Then:
If
E
is an rvalue andenable_borrowed_range<remove_cv_t<T>>
isfalse
,ranges::begin(E)
is ill-formed.Otherwise, if
T
is an array type andremove_all_extents_t<T>
is an incomplete type,ranges::begin(E)
is ill-formed with no diagnostic required.Otherwise, if T is an array type,
ranges::begin(E)
is expression-equivalent tot + 0
.Otherwise, if
auto(t.begin())
is a valid expression whose type modelsinput_or_output_iterator
,ranges::begin(E)
is expression-equivalent toauto(t.begin())
.Otherwise, if
T
is a class or enumeration type andauto(begin(t))
is a valid expression whose type modelsinput_or_output_iterator
with overload resolution performed in a context in which unqualified lookup forbegin
finds only the declarationsvoid begin(auto&) = delete; void begin(const auto&) = delete;
then
ranges::begin(E)
is expression-equivalent toauto(begin(t))
with overload resolution performed in the above context.Otherwise,
ranges::begin(E)
is ill-formed.
That is to say, it will not only perform specific operations on the array type but also decide whether to invoke member function range.begin()
or free function begin(range)
based on the validity of the expression, this already covers the behavior described by the so-called range-expression. And ranges::end
follows similar rules. So I think you can simply do
template<std::ranges::range T>
void f(T arg) {
for (auto&& e : arg) { // guaranteed to work
}
}
It should be noted that ranges::begin
requires that the returned type must model input_or_output_iterator
, and ranges::end
also requires that the returned type must model sentinel_for
type returned by ranges::begin
, so that T
is enough to be a range
. The range-expression does not have such constraints, it only checks the validity of the expression, so a minimal type that can use a range-based for loop could be
struct I {
int operator*();
I& operator++();
bool operator!=(const I&) const;
};
struct R {
I begin();
I end();
};
for (auto x : R{}) { } // well-formed
But I don't think you're interested in such a case since I
is not sufficient to constitute a valid iterator.