Скотт Мейерс - Эффективный и современный С++. 42 рекомендации по использованию С++11 и С++14
- Название:Эффективный и современный С++. 42 рекомендации по использованию С++11 и С++14
- Автор:
- Жанр:
- Издательство:Вильямс
- Год:2016
- Город:Москва
- ISBN:978-5-8459-2000-3
- Рейтинг:
- Избранное:Добавить в избранное
-
Отзывы:
-
Ваша оценка:
Скотт Мейерс - Эффективный и современный С++. 42 рекомендации по использованию С++11 и С++14 краткое содержание
В книге рассматриваются следующие темы. Освоение С++11 и С++14 — это больше, чем просто ознакомление с вводимыми этими стандартами возможностями (например, объявлениями типов
, семантикой перемещения, лямбда-выражениями или поддержкой многопоточности). Вопрос в том, как использовать их эффективно, чтобы создаваемые программы были корректны, эффективны и переносимы, а также чтобы их легко можно было сопровождать. Именно этим вопросам и посвящена данная книга, описывающая создание по-настоящему хорошего программного обеспечения с использованием C++11 и С++14 — т.е. с использованием современного С++.
■ Преимущества и недостатки инициализации с помощью фигурных скобок, спецификации
, прямой передачи и функций
интеллектуальных указателей
■ Связь между
,
, rvalue-ссылками и универсальными ссылками
■ Методы написания понятных, корректных,
лямбда-выражений
■ Чем
отличается от
, как они используются и как соотносятся с API параллельных вычислений С++
■ Какие из лучших методов “старого” программирования на С++ (т.е. С++98) должны быть пересмотрены при работе с современным С++
Более чем 20 лет книги
серии
являются критерием уровня книг по программированию на С++. Понятное пояснение сложного технического материала принесло ему всемирную известность. Он всегда самый желанный гость на международных конференциях, а его услуги консультанта широко востребованы во всем мире.
Скотт Мейерс Эффективный и современный С++, После изучения основ С++ я перешел к изучению того, как применять С++ в промышленном программировании, с помощью серии книг Скотта Мейерса Эффективный С++. Эффективный и современный С++ — наиболее важная из книг серии, предлагающая ключевые рекомендации, стили и идиомы, позволяющие эффективно использовать современный С++. Вы еще не купили эту книгу? Сделайте это прямо сейчас. Герб Саттер,
глава Комитета ISO по стандартизации С++, специалист в области архитектуры программного обеспечения на С++ в Microsoft
Эффективный и современный С++. 42 рекомендации по использованию С++11 и С++14 - читать онлайн бесплатно полную версию (весь текст целиком)
Интервал:
Закладка:
: Person(rhs) // конструктор; вызывает конструктор
{ … } // базового класса с прямой передачей!
SpecialPerson(SpecialPerson&& rhs) // Перемещающий
: Person(std::move(rhs))// конструктор; вызывает конструктор
{ … } // базового класса с прямой передачей!
};
Это тот же код, который вы видели ранее, в конце предыдущего раздела, включая комментарии, увы, оставшиеся справедливыми. Копируя или перемещая объект SpecialPerson
, мы ожидаем, что части базового класса будут скопированы или перемещены с помощью копирующего или, соответственно, перемещающего конструктора базового класса. Однако в этих функциях мы передаем объекты SpecialPerson
конструкторам базового класса, а поскольку SpecialPerson
не совпадает с Person
(даже после применения std::decay
), конструктор с универсальной ссылкой в базовом классе оказывается включенным и без проблем проходит проверку на идеальное совпадение с аргументом SpecialPerson
. Это точное соответствие лучше преобразования производного класса в базовый, необходимого для связывания объекта SpecialPerson
с параметром Person
в копирующем и перемещающем конструкторах класса Person
, так что при имеющемся коде копирование и перемещение объектов SpecialPerson
будет использовать для копирования и перемещения частей базового класса конструктор с универсальной ссылкой класса Person
! Это чудное ощущение дежавю раздела 5.4…
Производный класс просто следует обычным правилам реализации копирующего и перемещающего конструкторов производного класса, поэтому решение этой проблемы находится в базовом классе и, в частности, в условии, которое контролирует включение конструктора с универсальной ссылкой класса Person
. Теперь мы понимаем, что надо включать шаблонный конструктор не для любого типа аргумента, отличного от Person
, а для любого типа аргумента, отличного как от Person
, так и от типа, производного от Person
. Ох уж это наследование!
Вас уже не должно удивлять обилие всяческих полезных шаблонов в стандартной библиотеке, так что известие о наличии шаблона, который определяет, является ли один класс производным от другого, вы должны воспринять с полным спокойствием. Он называется std::is_base_of
. Значение std::is_base_of::value
истинно, если T2
— класс, производный от T1
. Пользовательские типы рассматриваются как производные от самих себя, так что std::is_base_of::value
истинно, если T
представляет собой пользовательский тип. (Если T
является встроенным типом, std::is_base_of::value
ложно.) Это удобно, поскольку мы хотим пересмотреть наше условие, управляющее отключением конструктора с универсальной ссылкой класса Person таким образом, чтобы этот конструктор был включен, только если тип T
после удаления всех ссылок и квалификаторов const
и volatile
не являлся ни типом Person
, ни классом, производным от Person
. Применение std::is_base_of
вместо std::is_same
дает нам то, что требуется:
class Person {
public:
template<
typename T,
typename = typename std::enable_if<
!std:: is_base_of
typename std::decay::type
>::value
>::type
>
explicit Person(T&& n);
};
Вот теперь работа завершена. Вернее, завершена при условии, что мы пишем код на С++11. При использовании С++14 этот код будет работать, но мы можем использовать псевдонимы шаблонов для std::enable_if
и std::decay
, чтобы избавиться от хлама в виде typename
и ::type
, получая несколько более приятный код:
class Person { // С++14
public:
template<
typename T,
typename = std::enable_if _t< // Меньше кода здесь
!std::is_base_of
std::decay _t // И здесь
>::value
> // И здесь
>
explicit Person(T&& n);
};
Ладно, я признаю: я соврал. Мы до сих пор не сделали всю работу. Но мы уже близки к завершению. Соблазнительно близки. Честно!
Мы видели, как использовать std::enable_if
для выборочного отключения конструктора с универсальной ссылкой класса Person
для типов аргументов, которые мы хотим обрабатывать с помощью копирующего и перемещающего конструкторов, но мы еще не видели, как применить его для того, чтобы отличать целочисленные аргументы от не являющихся таковыми. В конце концов, таковой была наша первоначальная цель; проблема неоднозначности конструктора была всего лишь неприятностью, подхваченной по дороге.
Все, что нам нужно сделать (и на этот раз “все” действительно означает, что это уже все), — это (1) добавить перегрузку конструктора Person
для обработки целочисленных аргументов и (2) сильнее ограничить шаблонный конструктор так, чтобы он был отключен для таких аргументов. Положите эти ингредиенты в кастрюлю со всем остальным, что мы уже обсуждали, варите на медленном огне и наслаждайтесь ароматом успеха:
class Person {
public:
template<
typename T,
typename = std::enable_if_t<
!std::is_base_of>::value
&&
!std::is_integral>::value
>
>
explicit Person(T&& n) // Конструктор для std::string и
: name(std::forward(n)) // аргументов, приводимых к
{ … } // std::string
explicit Person(int idx) // Конструктор для
: name(nameFromIdx(idx)) // целочисленных аргументов
{ … }
… // Копирующий и перемещающий конструкторы и т.д.
private:
std::string name;
};
Красота! Ну ладно, красота в основном для тех, кто фетишизирует метапрограммирование с использованием шаблонов, но факт остается фактом: этот подход не только справляется с работой, но и делает это с чрезвычайным апломбом. Применение прямой передачи предполагает высокую эффективность, а управление сочетанием универсальных ссылок и перегрузки вместо их запрета позволяет применять этот метод в обстоятельствах (таких, как разработка конструкторов), когда перегрузка неизбежна.
Первые три рассмотренные в данном разделе метода — отказ от перегрузки, передача const T&
и передача по значению — указывают тип каждого параметра в вызываемой функции или функциях. Последние два метода — диспетчеризация дескрипторов и ограничения шаблонов — используют прямую передачу, а следовательно, типы параметров не указывают. Это фундаментальное решение — указывать типы или нет — имеет свои следствия.
Как правило, прямая передача более эффективна, потому что позволяет избежать создания временных объектов исключительно с целью соответствия типу объявления параметра. В случае конструктора Person
прямая передача допускает передачу строкового литерала, такого как "Nancy"
, непосредственно в конструктор для std::string
внутри Person, в то время как методы, не использующие прямой передачи, вынуждены создавать временный объект std::string
из строкового литерала для удовлетворения спецификации параметра конструктора Person
.
Интервал:
Закладка: