In my first blog post i will show you my short journey into std::ranges
. Ranges were introduced in the C++20 standard and originated from the rangeV3 library.
In especially, the iteration over collections are super simple now. They also allow to iterate over the results of some member function. This is called projection. I also discuss two example where other capabilities as filtering and views in general are shown
Example 1: Projection onto member functions
Consider the following struct which has three member functions:
1 | struct Shape |
Now we have a vector of shapes and want to iterate over the vector but we want as quantity directly the return value of the member function, i.e. we have
1 | int main(){ |
This behavior can be accomplished with so called views
in std::ranges
. We need for this task std::ranges::views::transform
or the synonym std::ranges::transform_view
to which we can directly pass a pointer to a member function.
We can then write
1 | for(const auto area : std::ranges::transform_view(shapes, &Shape::getArea )) |
which can also be wrapped obviously into a function
1 | auto area(std::vector<Shape>& shapes) |
transform_view
is also suited for a more general approach. We can plugin anything invokable, e.g. lambdas as follows
1 | int i = 0; |
I think this approach pretty nice since it reduces a lot of noise around how to iterate over collections.
Using the lambda we can also call member functions which need arguments.
If we have the member function as follows
1 | struct Shape |
We can also use it in the following way
1 | int i = 0; |
Example 2: Iterate over a filtered subset of a collection
Another nice example is the feature to filter automatically the stuff which you are not interested in.
This can be done with std::ranges::filter_view
or again with the synonym std::ranges::views::filter
.
We have a vector of integers. We only want to iterate over the prime numbers. This can be done in the following old school way:
1 | bool isPrime(int n) |
Using the ranges library we can write
1 | int main() { |
or equivalently
1 | int main() { |
where the operator |
unites the predicate isPrime
with the vector v
.
I also benchmarked all three examples at quickbench using a vector with 1000 entries, see Figure below.
As you can see, there is almost no difference in the time taken. At least for the gcc10.2 compiler.
Nevertheless, if we change the predicate isPrime
to something trivial, e.g.
1 | bool isPrime(int n){ return true;} |
The benchmark result changes to quickbench.
Here, one can see that the by hand approach is much faster. In my opinion this is due to the inlining of the function in the byHand
approach.
Intrestingly, if i change the predicate slightly by introducing the function to be constexpr
.
1 | constexpr bool isPrime(int n) |
the results are again quite different and the ranges1
approach gains speed:
Example 3: Iterate over a view without storing any collection directly
In the example before i always wrote std::vector<int> v{0,1,2,3,4,5};
to define a vector over which we iterated. As you may have noticed, i used a different approach in the benchmark examples. With the ranges library you can construct views directly and never even have to store the complete vector. E.g. we can use std::ranges::iota_view
as follows
1 | auto v = std::ranges::iota_view{1, 1000}; |
or directly
1 | for (auto i : std::ranges::iota_view{1, 1000}) |
Bonus: Chaining stuff together
You can also chain operation together, e.g. if we want to iterate over the square of primes:
1 | int main() { |
At least to me these views seem to be really powerful.