Вызов функции remove_from_Folders() гарантирует отсутствие у объектов класса Folder указателей на удаленный объект класса Message. Компилятор автоматически вызывает деструктор класса string для освобождения объекта contents, а деструктор класса set освобождает память, используемую элементами набора.

Оператор присвоения копии класса Message

Как обычно, оператор присвоения и оператор присвоения копии класса Folder должны выполнять действия конструктора копий и деструктора. Как всегда, крайне важно структурировать свой код так, чтобы он выполнялся правильно, даже если операнды слева и справа — тот же объект.

В данном случае защита против присвоения самому себе осуществляется за счет удаления указателей на это сообщение из папок левого операнда прежде, чем вставить указатели в папки правого операнда:

Messages Message::operator=(const Message &rhs) {

 // отработать присвоение себе самому, удаляя указатели прежде вставки

 remove_from_Folders();   // обновить существующие папки

 contents = rhs.contents; // копировать содержимое сообщения из rhs

 folders = rhs.folders;   // копировать указатели Folder из rhs

 add_to_Folders(rhs);     // добавить это сообщение к данным папкам

 return *this;

}

Если левый и правый операнды — тот же объект, то у них тот же адрес. Если вызвать функцию remove_from_Folders() после вызова функции add_to_Folders(), это сообщение будет удалено изо всех соответствующих ему папок.

Функция swap() класса Message

Библиотека определяет версии функции swap() для классов string и set (см. раздел 9.2.5). В результате класс Message извлечет пользу из определения собственной версии функции swap(). При определении специфической для класса Message версии функции swap() можно избежать лишних копирований членов contents и folders.

Но наша функция swap() должна также управлять указателями Folder, которые указывают на обмениваемые сообщения. После такого вызова, как swap(m1, m2), указатели Folder, указывающие на объект m1, должны теперь указать на объект m2, и наоборот.

Для управления указателями Folder осуществляются два прохода по всем элементам folders. Первый проход удалит сообщения из соответствующих папок. Затем вызов функции swap() совершит обмен переменных-членов. Второй проход по элементам folders добавляет указатели на обмениваемые сообщения:

void swap(Message &lhs, Message &rhs) {

 using std::swap; // в данном случае не обязательно, но привычка

                  // хорошая

 // удалить указатели на каждое сообщение из их (оригинальных) папок

 for (auto f: lhs.folders)

  f->remMsg(&lhs);

 for (auto f: rhs.folders)

  f->remMsg(&rhs); // обмен наборов указателей contents и folders

 swap(lhs.folders, rhs.folders);   // использует swap(set&, set&)

 swap(lhs.contents, rhs.contents); // swap(string&, string&)

 // добавляет указатели на каждое сообщение в их (новые) папки

 for (auto f: lhs.folders)

  f->addMsg(&lhs);

 for (auto f: rhs.folders)

  f->addMsg(&rhs);

}

Упражнения раздела 13.4

Упражнение 13.33. Почему параметр функций-членов save() и remove() класса Message имеет тип Folder&? Почему этот параметр не определен как Folder или const Folder&?

Упражнение 13.34. Напишите класс Message, как описано в этом разделе.

Упражнение 13.35. Что случилось бы, используй класс Message синтезируемые версии функций-членов управления копированием?

Упражнение 13.36. Разработайте и реализуйте соответствующий класс Folder. Этот класс должен содержать набор указателей на сообщения в той папке.

Упражнение 13.37. Добавьте в класс Message функции-члены удаления и вставки заданного Folder* в folders. Эти члены аналогичны функциям-членам addMsg() и remMsg() класса Folder.

Упражнение 13.38. Для определения оператора присвоения класса Message не использовалась технология копирования и обмена. Почему, по вашему?

Перейти на страницу:

Похожие книги