Когда необходим контейнер, содержащий объекты, связанные наследованием, как правило, определяют контейнер указателей (предпочтительно интеллектуальных (см. раздел 12.1)) на базовый класс. Как обычно, динамический тип объекта, на который указывает этот указатель, мог бы быть типом базового класса или типом, производным от него:
vector
basket.push_back(make_shared
("0-201-82470-1", 50));
basket.push_back(
make_shared
//
//
cout << basket.back()->net_price(15) << endl;
Поскольку вектор basket
содержит указатели shared_ptr
, для получения объекта, функция net_price()
которого выполнится, следует обратиться к значению, возвращенному функцией basket.back()
. Для этого в вызове функции net_price()
используется оператор ->
. Как обычно, вызываемая версия функции net_price()
зависит от динамического типа объекта, на который указывает этот указатель.
Следует заметить, что вектор basket
был определен как shared_ptr
, все же во втором вызове функции push_back()
был передан указатель на объект класса Bulk_quote
. Подобно тому, как можно преобразовать обычный указатель на производный тип в указатель на тип базового класса (см. раздел 15.2.2), можно также преобразовать интеллектуальный указатель на производный тип в интеллектуальный указатель на тип базового класса. Таким образом, вызов функции make_shared
возвращает объект shared_ptr
, в который преобразуется shared_ptr
при вызове функции push_back()
. В результате, несмотря на внешний вид, у всех элементов вектора basket
будет тот же тип.
Упражнение 15.28. Определите вектор для содержания объектов класса Quote
, но поместите в него объекты класса Bulk_quote
. Вычислите общую сумму результатов вызова функции net_price()
для всех элементов вектора.
Упражнение 15.29. Повторите предыдущую программу, но на сей раз храните указатели shared_ptr
на объекты типа Quote
. Объясните различие в сумме данной версии программы и предыдущей. Если никакой разницы нет, объясните почему.
Basket
Ирония объектно-ориентированного программирования на языке С++ в том, что невозможно использовать объекты непосредственно. Вместо них приходится использовать указатели и ссылки. Поскольку указатели усложняют программы, зачастую приходится определять вспомогательные классы, чтобы избежать осложнений. Для начала определим класс, представляющий корзину покупателя:
class Basket {
public:
//
//
void add_item(const std::shared_ptr
&sale)
{ items.insert(sale); }
//
//
double total_receipt(std::ostream&) const;
private:
//
//
static bool compare(const std::shared_ptr
&lhs,
const std::shared_ptr
&rhs)
{ return lhs->isbn() < rhs->isbn(); }
//
//
std::multiset
items{compare};
}
Для хранения транзакций класс использует контейнер multiset
(см. раздел 11.2.1), позволяющий содержать несколько транзакций по той же книге, чтобы все транзакции для данной книги находились вместе (см. раздел 11.2.2).
Элементами контейнера multiset
будут указатели shared_ptr
, и для них нет оператора "меньше". В результате придется предоставить собственный оператор сравнения для упорядочивания элементов (см. раздел 11.2.2). Здесь определяется закрытая статическая функция-член compare()
, сравнивающая isbn
объектов, на которые указывают указатели shared_ptr
. Инициализируем контейнер multiset
с использованием этой функции сравнения и внутриклассового инициализатора (см. раздел 7.3.1):