Существует ли istream const итератор? - PullRequest
0 голосов
/ 22 апреля 2020

Существует ли такая вещь, как итератор std :: istream const?

Следующий код не будет компилироваться, поскольку std::istream_iterator в foo() не может привязаться к ссылке const std::istream на временный объект, созданный в main().

// main.cpp
#include <iostream>
#include <sstream>
#include <iterator>

void foo( const std::istream& s )
{
  std::istream_iterator<char> i( s );
  // No std::istream_const_iterator, but anything by another name?
  std::cout << *i;
}

int main( int argc, char* argv[] )
{
  std::string p( "abcdefghijklmnopqrstuvwxyz" );
  foo( std::stringstream(p) );

  return 0;
}

Существует ли istream итератор, который может связываться с константой istream?

Только что задав этот вопрос , я только что узнал, что istream_iterator инициализирует ctors и выполняет первое чтение . Думаю, мне все еще не ясно: это чтение обязательно изменяет границу istream? Если нет, то, похоже, должен быть какой-то константный итератор, который может связываться с const istream, нет?

$ g++ --version
g++ (GCC) 9.2.1 20190827 (Red Hat 9.2.1-1)
Copyright (C) 2019 Free Software Foundation, Inc.
This is free software; see the source for copying conditions.  There is NO
warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.

$ g++ -g ./main.cpp && ./a.out
./main.cpp: In function ‘void foo(const istream&)’:
./main.cpp:84:34: error: binding reference of type ‘std::istream_iterator<char>::istream_type&’ {aka ‘std::basic_istream<char>&’} to ‘const istream’ {aka ‘const std::basic_istream<char>’} discards qualifiers
   84 |   std::istream_iterator<char> i( s );
      |                                  ^
In file included from /usr/include/c++/9/iterator:66,
                 from ./main.cpp:80:
/usr/include/c++/9/bits/stream_iterator.h:68:38: note:   initializing argument 1 of ‘std::istream_iterator<_Tp, _CharT, _Traits, _Dist>::istream_iterator(std::istream_iterator<_Tp, _CharT, _Traits, _Dist>::istream_type&) [with _Tp = char; _CharT = char; _Traits = std::char_traits<char>; _Dist = long int; std::istream_iterator<_Tp, _CharT, _Traits, _Dist>::istream_type = std::ba
sic_istream<char>]’
   68 |       istream_iterator(istream_type& __s)
      |                        ~~~~~~~~~~~~~~^~~

1 Ответ

2 голосов
/ 22 апреля 2020

означает ли это, что чтение обязательно изменяет границу istream?

Да.

istream_iterator - это удобный класс, который позволяет обрабатывать istream объекты как будто они являются контейнерами, такими как std::vector или массив.

Внизу istream - это объект, используемый для чтения из потока. И да, чтение из потока изменяет istream. Иначе как istream будет отслеживать внутреннее состояние, чтобы указать, была ли попытка чтения успешной или нет, сколько символов было прочитано и т. Д. c.?

Так как вам нужен не const istream объекты для чтения, нет смысла иметь возможность создавать istream_iterator из const istream объектов.

...