Я создал минимальный рабочий пример, чтобы показать проблему, с которой я столкнулся при использовании итераторов STL. Я использую istream_iterator
для чтения floats
s (или других типов) из std::istream
:
#include <iostream>
#include <iterator>
#include <algorithm>
int main() {
float values[4];
std::copy(std::istream_iterator<float>(std::cin), std::istream_iterator<float>(), values);
std::cout << "Read exactly 4 floats" << std::endl; // Not true!
}
Это читает все возможные floats
s, до EOF в values
, который имеет фиксированный размер, 4, поэтому теперь ясно, что я хочу ограничить диапазон, чтобы избежать переполнения и прочитать точно/не более 4 значений.
С более «обычными» итераторами (например, RandomAccessIterator), при условии, что begin+4
не превышает конца, который вы бы сделали:
std::copy(begin, begin+4, out);
Чтобы прочитать ровно 4 элемента.
Как это сделать с std::istream_iterator
? Очевидная идея состоит в том, чтобы изменить вызов std::copy
на:
std::copy(std::istream_iterator<float>(std::cin), std::istream_iterator<float>(std::cin)+4, values);
Но (довольно предсказуемо) это не компилируется, кандидатов на operator+
нет:
g++ -Wall -Wextra test.cc
test.cc: In function ‘int main()’:
test.cc:7: error: no match for ‘operator+’ in ‘std::istream_iterator<float, char, std::char_traits<char>, long int>(((std::basic_istream<char, std::char_traits<char> >&)(& std::cin))) + 4’
Какие-либо предложения? Есть ли правильный способ "STLified" pre-C++0x для достижения этого? Очевидно, что я мог просто написать это как цикл for, но я хочу узнать кое-что о STL здесь. Я наполовину задавался вопросом о злоупотреблении std::transform
или std::merge
и т. д., чтобы каким-то образом добиться этой функциональности, но я не совсем понимаю, как это сделать.