WedX - журнал о программировании и компьютерных науках

Перегрузка оператора C++ — перегрузка оператора вывода ‹‹

Я только начинаю изучать базовый синтаксис С++, и меня немного смущает фрагмент кода, с которым я столкнулся.

Для класса, созданного с именем MyString, была определена перегрузка оператора как:

ostream& operator<<(ostream& os, const MyString& s)
{
    os << s.data;
    return os;
}

Затем в некоторой функции драйвера оператор:

cout << s3 << endl;

Выполняется, где s3 имеет тип объекта MyString. Результат выводит значение s3.

Я не совсем понимаю, как работает это утверждение. После возни с ним кажется, что конструктор копирования вызывается один раз, а затем деконструируются 3 объекта. Как именно работает эта линия? Похоже, что оператор принимает ссылку на ostream и MyString, но тоже не endl? Кроме того, почему может быть вызван только 1 конструктор копирования, когда используются два экземпляра «‹‹»? Может быть, я даже не задаю правильных вопросов, или мои вопросы даже не имеют смысла, потому что я действительно сбит с толку тем, что происходит в этих строках. Если это так, может ли кто-нибудь изложить просто общее объяснение того, что происходит?


  • По какой книге C++ вы учитесь? 06.12.2011
  • это слишком широко, чтобы ответить здесь, вам нужно обратиться к хорошей книге. 06.12.2011
  • C++ Primer, 4-е изд., Липпман 06.12.2011
  • Вы называете этот базовый синтаксис С++? Кроме того, перегрузка операторов, когда вы только начали изучать С++ :)? 06.12.2011
  • Если вы хотите, чтобы мы объяснили, сколько существует копий экземпляров MyString, вам потребуется предоставить более полный пример кода. Мы не можем диагностировать эту проблему без полного примера. 06.12.2011
  • Я думаю, это на самом деле три вопроса: а) что такое вызов по ссылке? б) что такое перегрузка оператора? c) что такое приоритет оператора? - много информации, однако 06.12.2011

Ответы:


1

Это очень общий вопрос, но я постараюсь прояснить ваше недопонимание.

Когда вы говорите ostream& operator<<(ostream& os, const MyString& s) { ... }, вы просто определяете функцию, которая принимает ostream& в качестве первого параметра и const Mystring& в качестве второго параметра и возвращает ostream&. Функция называется operator<< и может быть вызвана с помощью сокращенного синтаксиса x << y для вызова operator<<(x, y).

Когда вы делаете cout << s3 << endl;, это то же самое, что и operator<<(operator<<(cout, s3), endl);.

Ни конструктор копирования, ни деструктор для MyString не вызываются из этого кода. Сообщения, которые вы видите, откуда-то еще.

05.12.2011

2

Можно немного разбить:

cout // this is the ostream your inserting to (stdout)
  << s3 // this calls your defined operator that writes s.data
  << endl; // this calls the operator<< for std::endl

ostream& operator<<(ostream& os, const MyString& s)
{
    // here os is the ostream (stdout) you're using via cout
    // s is s3 that you passed in
    os << s.data; // this calls operator<< for data
    return os; // this returns the reference so the subsequent call to << endl can append to the stream
}
05.12.2011

3

Конструктор копирования MyString не требуется для вызова этого оператора, поскольку MyString передается как ссылка на пользовательский оператор‹‹ для MyString.

Endl не отправляется пользовательскому оператору‹‹ для MyString.

05.12.2011
  • Значит, s3 ‹‹ endl не использует перегруженный оператор, а cout ‹‹ s3? Это потому, что s3 ‹‹ endl не соответствует требуемым параметрам ostream&/MyString&? 06.12.2011
  • Нет s3‹‹endl -- это ((cout ‹‹ s3) ‹‹ endl) --- и результатом первого оператора является cout. 06.12.2011

  • 4

    Я предполагаю, что конструктор копирования может быть сгенерирован вашим компилятором для cout. Поток — это довольно сложный объект, но выбор C++ заключается в том, чтобы предоставить такой синтаксический сахар (например, встроенные конструкторы), чтобы абстрагироваться от некоторых деталей. В других ответах уже указывалось на существенный момент, что при использовании ссылок копирование не выполняется для передачи параметров.

    05.12.2011

    5

    Правила языка гласят, что если функция, не являющаяся членом (оператор‹‹, является функцией, не являющейся членом класса MyString), определена в том же пространстве имен, что и MyString, то эта функция, не являющаяся членом, может быть выбрана компилятором для разрешения этого вызова (как объяснил один из респондентов) Я не использую здесь стандартную терминологию, если у кого-то есть более точное описание этого принципа, сообщите нам.

    05.12.2011
  • У меня был непосредственный опыт такого поведения, когда я переключился с MSVC2003 на MSVC2005: у меня был код вроде ofstream(log) ‹‹ здесь..\n. После переключения вместо сообщения у меня был мусор (значение указателя) в файле. 06.12.2011
  • Новые материалы

    Объяснение документов 02: BERT
    BERT представил двухступенчатую структуру обучения: предварительное обучение и тонкая настройка. Во время предварительного обучения модель обучается на неразмеченных данных с помощью..

    Как проанализировать работу вашего классификатора?
    Не всегда просто знать, какие показатели использовать С развитием глубокого обучения все больше и больше людей учатся обучать свой первый классификатор. Но как только вы закончите..

    Работа с цепями Маркова, часть 4 (Машинное обучение)
    Нелинейные цепи Маркова с агрегатором и их приложения (arXiv) Автор : Бар Лайт Аннотация: Изучаются свойства подкласса случайных процессов, называемых дискретными нелинейными цепями Маркова..

    Crazy Laravel Livewire упростил мне создание электронной коммерции (панель администратора и API) [Часть 3]
    Как вы сегодня, ребята? В этой части мы создадим CRUD для данных о продукте. Думаю, в этой части я не буду слишком много делиться теорией, но чаще буду делиться своим кодом. Потому что..

    Использование машинного обучения и Python для классификации 1000 сезонов новичков MLB Hitter
    Чему может научиться машина, глядя на сезоны новичков 1000 игроков MLB? Это то, что исследует это приложение. В этом процессе мы будем использовать неконтролируемое обучение, чтобы..

    Учебные заметки: создание моего первого пакета Node.js
    Это мои обучающие заметки, когда я научился создавать свой самый первый пакет Node.js, распространяемый через npm. Оглавление Глоссарий I. Новый пакет 1.1 советы по инициализации..

    Забудьте о Matplotlib: улучшите визуализацию данных с помощью умопомрачительных функций Seaborn!
    Примечание. Эта запись в блоге предполагает базовое знакомство с Python и концепциями анализа данных. Привет, энтузиасты данных! Добро пожаловать в мой блог, где я расскажу о невероятных..


    Для любых предложений по сайту: wedx@cp9.ru