Стенли Липпман - Язык программирования C++. Пятое издание
- Название:Язык программирования C++. Пятое издание
- Автор:
- Жанр:
- Издательство:Издательский дом Вильямс
- Год:2014
- Город:Москва
- ISBN:978-5-8459-1839-0
- Рейтинг:
- Избранное:Добавить в избранное
-
Отзывы:
-
Ваша оценка:
Стенли Липпман - Язык программирования C++. Пятое издание краткое содержание
Вы держите в руках новое издание популярного и исчерпывающего бестселлера по языку программирования С++, которое было полностью пересмотрено и обновлено под
. Оно поможет вам быстро изучить язык и использовать его весьма эффективными и передовыми способами. В соответствии с самыми передовыми и современными методиками изложения материала авторы демонстрируют использование базового языка и его стандартной библиотеки для разработки эффективного, читабельного и мощного кода.
С самого начала этой книги читатель знакомится со стандартной библиотекой С++, ее самыми популярными функциями и средствами, что позволяет сразу же приступить к написанию полезных программ, еще не овладев всеми нюансами языка. Большинство примеров из книги было пересмотрено так, чтобы использовать новые средства языка и продемонстрировать их наилучшие способы применения. Эта книга — не только проверенное руководство для новичков в С++, она содержит также авторитетное обсуждение базовых концепций и методик языка С++ и является ценным ресурсом для опытных программистов, особенно желающих побыстрей узнать об усовершенствованиях С++11.
Стенли Б. Липпман Жози Лажойе Барбара Э. Му • Узнайте, как использовать новые средства языка С++11 и стандартной библиотеки для быстрого создания надежных программ, а также ознакомьтесь с высокоуровневым программированием
• Учитесь на примерах, в которых показаны передовые стили программирования и методики проектирования
• Изучите принципы и узнайте почему язык С++11 работает именно так
• Воспользуйтесь множеством перекрестных ссылок, способных помочь вам объединить взаимосвязанные концепции и проникнуть в суть
• Ознакомьтесь с современными методиками обучения и извлеките пользу из упражнений, в которых подчеркиваются ключевые моменты, позволяющие избежать проблем
• Освойте лучшие методики программирования и закрепите на практике изученный материал
Исходный код примеров можно загрузить с веб-страницы книги на сайте издательства по адресу: http://www.williamspublishing.com
Язык программирования C++. Пятое издание - читать онлайн бесплатно полную версию (весь текст целиком)
Интервал:
Закладка:
vector> basket;
basket.push_back(make_shared("0-201-82470-1", 50));
basket.push_back(
make_shared("0-201-54848-8", 50, 10, .25));
// вызов версии, определенной в Quote, выводит 562.5,
// т.е. со скидкой, меньше, чем 15 * $50
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
. Объясните различие в сумме данной версии программы и предыдущей. Если никакой разницы нет, объясните почему.
15.8.1. Разработка класса
Basket
Ирония объектно-ориентированного программирования на языке С++ в том, что невозможно использовать объекты непосредственно. Вместо них приходится использовать указатели и ссылки. Поскольку указатели усложняют программы, зачастую приходится определять вспомогательные классы, чтобы избежать осложнений. Для начала определим класс, представляющий корзину покупателя:
class Basket {
public:
// Basket использует синтезируемый стандартный конструктор и
// функции-члены управления копированием
void add_item(const std::shared_ptr &sale)
{ items.insert(sale); }
// выводит общую стоимость каждой книги и общий счет для всех
// товаров в корзинке
double total_receipt(std::ostream&) const;
private:
// функция сравнения shared_ptr, необходимая элементам
// набора multiset
static bool compare(const std::shared_ptr &lhs,
const std::shared_ptr &rhs)
{ return lhs->isbn() < rhs->isbn(); }
// набор multiset содержит несколько стратегий расценок,
// упорядоченных по сравниваемому элементу
std::multiset, decltype(compare)*>
items{compare};
}
Для хранения транзакций класс использует контейнер multiset
(см. раздел 11.2.1), позволяющий содержать несколько транзакций по той же книге, чтобы все транзакции для данной книги находились вместе (см. раздел 11.2.2).
Элементами контейнера multiset
будут указатели shared_ptr
, и для них нет оператора "меньше". В результате придется предоставить собственный оператор сравнения для упорядочивания элементов (см. раздел 11.2.2). Здесь определяется закрытая статическая функция-член compare()
, сравнивающая isbn
объектов, на которые указывают указатели shared_ptr
. Инициализируем контейнер multiset
с использованием этой функции сравнения и внутриклассового инициализатора (см. раздел 7.3.1):
// набор multiset содержит несколько стратегий расценок,
// упорядоченных по сравниваемому элементу
std::multiset, decltype(compare)*>
items{compare};
Это объявление может быть трудно понять, но, читая его слева направо, можно заметить, что определяется контейнер multiset
указателей shared_ptr
на объекты класса Quote
. Для упорядочивания элементов контейнер multiset
будет использовать функцию с тем же типом, что и функция-член compare()
. Элементами контейнера multiset
будут объекты items
, которые инициализируются для использования функции compare()
.
Basket
Класс Basket
определяет только две функции. Функция-член add_item()
определена в классе. Она получает указатель shared_ptr
на динамически созданный объект класса Quote
и помещает его в контейнер multiset
. Вторая функция-член, total_receipt()
, выводит полученный счет для содержимого корзины и возвращает цену за все элементы в ней:
double Basket::total_receipt(ostream &os) const {
double sum = 0.0; // содержит текущую сумму
// iter ссылается на первый элемент в пакете элементов с тем же ISBN
// upper_bound() возвращает итератор на элемент сразу после
// конца этого пакета
for (auto iter = items.cbegin();
iter != items.cend();
iter = items.upper_bound(*iter)) {
// известно, что в Basket есть по крайней мере один элемент
// с этим ключом
// вывести строку для элемента этой книги
sum += print_total(os, **iter, items.count(*iter));
}
os << "Total Sale: " << sum << endl; // вывести в конце общий счет
return sum;
}
Цикл for
начинается с определения и инициализации итератора iter
на первый элемент контейнера multiset
. Условие проверяет, не равен ли iter
значению items.cend()
. Если да, то обработаны все покупки и цикл for
завершается. В противном случае обрабатывается следующая книга.
Интересный момент — выражение "инкремента" в цикле for
. Это не обычный цикл, читающий каждый элемент и перемещающий итератор iter
на следующий. При вызове функции upper_bound()
(см. раздел 11.3.5) он перескакивает через все элементы, которые соответствуют текущему ключу. Вызов функции upper_bound()
возвращает итератор на элемент сразу после последнего с тем же ключом, что и iter
. Возвращаемый итератор обозначает или конец набора, или следующую книгу.
Для вывода подробностей по каждой книге в корзине в цикле for
происходит вызов функции print_total()
(см. раздел 15.1):
sum += print_total(os, **iter, items.count(*iter));
Аргументами функции print_total()
являются поток ostream
для записи, обрабатываемый объект Quote
и счет. При обращении к значению итератора iter
возвращается указатель shared_ptr
, указывающий на объект, который предстоит вывести. Чтобы получить этот объект, следует обратиться к значению этого указателя shared_ptr
. Таким образом, выражение **iter
возвращает объект класса Quote
(или класса производного от него). Для выяснения количества элементов в контейнере multiset
с тем же ключом (т.е. с тем же ISBN) используется его функция-член count()
(см. раздел 11.3.5).
Интервал:
Закладка: