Альтернативная версия find_if, которая находит все, а не только первое?
Вопрос
Есть ли альтернативная версия std::find_if
который возвращает итератор по всем найденным элементам, а не только по первому?
Пример:
bool IsOdd (int i) {
return ((i % 2) == 1);
}
std::vector<int> v;
v.push_back(1);
v.push_back(2);
v.push_back(3);
v.push_back(4);
std::vector<int>::iterator it = find_if(v.begin(), v.end(), IsOdd);
for(; it != v.end(); ++it) {
std::cout << "odd: " << *it << std::endl;
}
Решение
Вы можете просто использовать for
петля:
for (std::vector<int>:iterator it = std::find_if(v.begin(), v.end(), IsOdd);
it != v.end();
it = std::find_if(++it, v.end(), IsOdd))
{
// ...
}
В качестве альтернативы вы можете поместить свое условие и действие в функтор (выполнение действия только в том случае, если условие истинно) и просто использовать std::foreach
.
Другие советы
в STL нет, но boost предлагает следующую функциональность:
Сначала всегда пытайтесь придумать типичное использование STL, вы также можете пойти на повышение.Вот более упрощенная форма из вышеупомянутого ответа Чарльза.
vec_loc = find_if(v3.begin(), v3.end(), isOdd);
if (vec_loc != v3.end())
{
cout << "odd elem. found at " << (vec_loc - v3.begin()) << "and elem found is " << *vec_loc << endl;
++vec_loc;
}
for (;vec_loc != v3.end();vec_loc++)
{
vec_loc = find_if(vec_loc, v3.end(), isOdd);
if (vec_loc == v3.end())
break;
cout << "odd elem. found at " << (vec_loc - v3.begin()) << "and elem found is " << *vec_loc << endl;
}
Не связан с StackOverflow