Випадок використання виник, коли хотіли зробити умовну копію (1. виконуване за допомогою copy_if
), але з контейнера значень у контейнер покажчиків на ці значення (2. виконуваний за допомогою transform
).
За допомогою доступних інструментів я не можу зробити це менш ніж за два кроки:
#include <vector>
#include <algorithm>
using namespace std;
struct ha {
int i;
explicit ha(int a) : i(a) {}
};
int main()
{
vector<ha> v{ ha{1}, ha{7}, ha{1} }; // initial vector
// GOAL : make a vector of pointers to elements with i < 2
vector<ha*> ph; // target vector
vector<ha*> pv; // temporary vector
// 1.
transform(v.begin(), v.end(), back_inserter(pv),
[](ha &arg) { return &arg; });
// 2.
copy_if(pv.begin(), pv.end(), back_inserter(ph),
[](ha *parg) { return parg->i < 2; }); // 2.
return 0;
}
Звичайно , ми могли б назвати remove_if
на pv
і усунути необхідність тимчасового, ще краще , хоча, це не складно реалізувати (для одинарних операцій) що - щось на зразок цього:
template <
class InputIterator, class OutputIterator,
class UnaryOperator, class Pred
>
OutputIterator transform_if(InputIterator first1, InputIterator last1,
OutputIterator result, UnaryOperator op, Pred pred)
{
while (first1 != last1)
{
if (pred(*first1)) {
*result = op(*first1);
++result;
}
++first1;
}
return result;
}
// example call
transform_if(v.begin(), v.end(), back_inserter(ph),
[](ha &arg) { return &arg; }, // 1.
[](ha &arg) { return arg.i < 2; });// 2.
- Чи є більш елегантне рішення із наявними стандартними інструментами бібліотеки C ++?
- Чи є причина, чому
transform_if
не існує в бібліотеці? Чи є поєднання існуючих інструментів достатнім обхідним шляхом та / або вважається ефективністю розумного поведінки?
transform_if
передбачає "перетворювати лише в тому випадку, якщо певний присудок задоволений". Більш описова назва того, що ви хочете, будеcopy_if_and_transform
!