Skip to content

Commit a648db3

Browse files
committed
Implement iterate algorithms
Problem: - There is no algorithm that generalizes `std::iota`. Filling a range 2 elements at a time is hard, more complex patterns are increasingly difficult. Solution: - Implement `iterate` and `iterate_n` which fill a range based on the execution of a binary operation which takes the previous value.
1 parent dbf91f4 commit a648db3

File tree

4 files changed

+186
-0
lines changed

4 files changed

+186
-0
lines changed

doc/algorithm.qbk

+10
Original file line numberDiff line numberDiff line change
@@ -224,6 +224,16 @@ Write a sequence of n increasing values to an output iterator
224224
Raise a value to an integral power ([^constexpr] since C++14)
225225
[endsect:power]
226226

227+
[section:iterate iterate ]
228+
[*[^[link header.boost.algorithm.iterate_hpp iterate] ] ]
229+
Store the result of a unary function to each element of a range. This is a generalization of iota.
230+
[endsect:iterate]
231+
232+
[section:iterate iterate ]
233+
[*[^[link header.boost.algorithm.iterate_hpp iterate] ] ]
234+
Store the result of a unary function to n element of a range. This is a generalization of iota.
235+
[endsect:iterate]
236+
227237
[endsect:misc_inner_algorithms]
228238

229239
[endsect:Misc]

include/boost/algorithm/iterate.hpp

+43
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,43 @@
1+
/*
2+
Copyright (c) Jonathan Gopel 2022.
3+
4+
Distributed under the Boost Software License, Version 1.0. (See accompanying
5+
file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
6+
*/
7+
8+
#ifndef BOOST_ALGORITHM_ITERATE_HPP
9+
#define BOOST_ALGORITHM_ITERATE_HPP
10+
11+
#include <boost/config.hpp>
12+
13+
#include <utility>
14+
15+
namespace boost {
16+
namespace algorithm {
17+
18+
template <class OutputIterator, class TInit, class UnaryOperation>
19+
BOOST_CXX14_CONSTEXPR TInit iterate(OutputIterator first, OutputIterator last,
20+
TInit init,
21+
UnaryOperation op) // TODO: BOOST_NOEXCEPT?
22+
{
23+
for (; first != last; ++first, init = op(init)) {
24+
*first = init;
25+
}
26+
return init;
27+
}
28+
29+
template <class OutputIterator, class Size, class TInit, class UnaryOperation>
30+
BOOST_CXX14_CONSTEXPR std::pair<OutputIterator, TInit>
31+
iterate(OutputIterator first, Size n, TInit init,
32+
UnaryOperation op) // TODO: BOOST_NOEXCEPT?
33+
{
34+
for (; n > 0; n -= 1, ++first, init = op(init)) {
35+
*first = init;
36+
}
37+
return std::make_pair(first, init);
38+
}
39+
40+
} // namespace algorithm
41+
} // namespace boost
42+
43+
#endif // BOOST_ALGORITHM_ITERATE_HPP

test/Jamfile.v2

+2
Original file line numberDiff line numberDiff line change
@@ -90,6 +90,8 @@ alias unit_test_framework
9090
# Find tests
9191
[ run find_not_test.cpp unit_test_framework : : : : find_not_test ]
9292
[ run find_backward_test.cpp unit_test_framework : : : : find_backward_test ]
93+
# Iterate tests
94+
[ run iterate_test.cpp unit_test_framework : : : : iterate_test ]
9395
;
9496
}
9597

test/iterate_test.cpp

+131
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,131 @@
1+
/*
2+
Copyright (c) Jonathan Gopel 2022.
3+
4+
Distributed under the Boost Software License, Version 1.0. (See accompanying
5+
file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
6+
*/
7+
8+
#include <boost/algorithm/iterate.hpp>
9+
10+
#include "iterator_test.hpp"
11+
12+
#include <boost/config.hpp>
13+
14+
#define BOOST_TEST_MAIN
15+
#include <boost/test/unit_test.hpp>
16+
17+
#include <boost/algorithm/cxx14/equal.hpp>
18+
19+
#include <algorithm>
20+
#include <array>
21+
#include <functional>
22+
#include <vector>
23+
24+
typedef int data_t;
25+
26+
static const int NUM_ELEMENTS = 4;
27+
28+
BOOST_CONSTEXPR data_t add_one(const data_t value) { return value + 1; }
29+
30+
BOOST_CONSTEXPR data_t add_two(const data_t value) { return value + 2; }
31+
32+
void test_iterate() {
33+
{ // iota
34+
std::vector<data_t> actual(NUM_ELEMENTS);
35+
36+
boost::algorithm::iterate(actual.begin(), actual.end(), 15, add_one);
37+
38+
std::vector<data_t> expected(NUM_ELEMENTS);
39+
expected.at(0) = 15;
40+
expected.at(1) = 16;
41+
expected.at(2) = 17;
42+
expected.at(3) = 18;
43+
BOOST_CHECK(actual.size() == expected.size());
44+
BOOST_CHECK(std::equal(actual.begin(), actual.end(), expected.begin()));
45+
}
46+
{ // striding
47+
std::vector<data_t> actual(NUM_ELEMENTS);
48+
49+
boost::algorithm::iterate(actual.begin(), actual.end(), 15, add_two);
50+
51+
std::vector<data_t> expected(NUM_ELEMENTS);
52+
expected.at(0) = 15;
53+
expected.at(1) = 17;
54+
expected.at(2) = 19;
55+
expected.at(3) = 21;
56+
BOOST_CHECK(std::equal(actual.begin(), actual.end(), expected.begin()));
57+
}
58+
}
59+
60+
void test_iterate_n() {
61+
{ // iota
62+
std::vector<data_t> actual(NUM_ELEMENTS);
63+
64+
boost::algorithm::iterate(actual.begin(), actual.size(), 15, add_one);
65+
66+
std::vector<data_t> expected(NUM_ELEMENTS);
67+
expected.at(0) = 15;
68+
expected.at(1) = 16;
69+
expected.at(2) = 17;
70+
expected.at(3) = 18;
71+
BOOST_CHECK(actual.size() == expected.size());
72+
BOOST_CHECK(std::equal(actual.begin(), actual.end(), expected.begin()));
73+
}
74+
{ // striding
75+
std::vector<data_t> actual(NUM_ELEMENTS);
76+
77+
boost::algorithm::iterate(actual.begin(), actual.size(), 15, add_two);
78+
79+
std::vector<data_t> expected(NUM_ELEMENTS);
80+
expected.at(0) = 15;
81+
expected.at(1) = 17;
82+
expected.at(2) = 19;
83+
expected.at(3) = 21;
84+
BOOST_CHECK(actual.size() == expected.size());
85+
BOOST_CHECK(std::equal(actual.begin(), actual.end(), expected.begin()));
86+
}
87+
}
88+
89+
BOOST_CXX14_CONSTEXPR inline bool constexpr_test_iterate() {
90+
int actual[] = {0, 0, 0, 0};
91+
92+
boost::algorithm::iterate(input_iterator<int *>(actual),
93+
input_iterator<int *>(actual + NUM_ELEMENTS), 15,
94+
add_one);
95+
96+
int expected[] = {15, 16, 17, 18};
97+
return boost::algorithm::equal(
98+
input_iterator<int *>(actual),
99+
input_iterator<int *>(actual + NUM_ELEMENTS),
100+
input_iterator<int *>(expected),
101+
input_iterator<int *>(expected + NUM_ELEMENTS));
102+
}
103+
104+
BOOST_CXX14_CONSTEXPR inline bool constexpr_test_iterate_n() {
105+
int actual[] = {0, 0, 0, 0};
106+
107+
boost::algorithm::iterate(input_iterator<int *>(actual), NUM_ELEMENTS, 15,
108+
add_one);
109+
110+
int expected[] = {15, 16, 17, 18};
111+
return boost::algorithm::equal(
112+
input_iterator<int *>(actual),
113+
input_iterator<int *>(actual + NUM_ELEMENTS),
114+
input_iterator<int *>(expected),
115+
input_iterator<int *>(expected + NUM_ELEMENTS));
116+
}
117+
118+
BOOST_AUTO_TEST_CASE(test_main) {
119+
test_iterate();
120+
test_iterate_n();
121+
122+
{
123+
BOOST_CXX14_CONSTEXPR bool result = constexpr_test_iterate();
124+
BOOST_CHECK(result);
125+
}
126+
127+
{
128+
BOOST_CXX14_CONSTEXPR bool result = constexpr_test_iterate_n();
129+
BOOST_CHECK(result);
130+
}
131+
}

0 commit comments

Comments
 (0)