Скотт Мейерс - Эффективный и современный С++. 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 - читать онлайн бесплатно полную версию (весь текст целиком)
Интервал:
Закладка:
class Widget { // В файле "widget.h"
public:
Widget();
… // Нет объявлений деструктора
// и перемещающих операций
private:
struct Impl;
std::shared_ptr pImpl; // std::shared_ptr
}; // вместо std::unique_ptr
и приведенном далее коде клиента, который включает заголовочный файл widget.h
Widget w1;
auto w2(std::move(w1)); // Перемещающее конструирование w2
w1 = std::move(w2); // Перемещающее присваивание w1
все компилировалось бы и работало именно так, как мы рассчитывали: w1
был бы создан конструктором по умолчанию, его значение было бы перемещено в w2
, а затем это значение, в свою очередь, было бы перемещено в w1
, после чего и w1
, и w2
были бы уничтожены (тем самым приводя к уничтожению объекта Widget::Impl
).
Различие в поведении указателей std::unique_ptr
и std::shared_ptr
для pImpl
вытекает из различий путей, которыми эти интеллектуальные указатели поддерживают пользовательские удалители. Для std::unique_ptr
тип удалителя является частью типа интеллектуального указателя, и это позволяет компилятору генерировать меньшие структуры данных времени выполнения и более быстрый код. Следствием этой более высокой эффективности является то, что указываемые типы должны быть полными, когда используются специальные функции-члены, генерируемые компиляторами (например, деструкторы или перемещающие операции). В случае std::shared_ptr
тип удалителя не является частью типа интеллектуального указателя. Это требует больших структур данных времени выполнения и несколько более медленного кода, но зато указываемые типы не обязаны быть полными при применении специальных функций-членов, генерируемых компиляторами.
При применении идиомы Pimpl в действительности нет никакого компромисса между характеристиками std::unique_ptr
и std::shared_ptr
, поскольку отношения между классами наподобие Widget
и Widget::Impl
представляют собой исключительное владение, и это делает единственно верным выбором в качестве инструмента интеллектуальный указатель std::unique_ptr
. Тем не менее стоит знать, что в других ситуациях — ситуациях, в которых осуществляется совместное владение (а следовательно, правильным выбором является std::shared_ptr
), — нет необходимости прыгать через горящие обручи определений функций, которую влечет за собой применение std::unique_ptr
.
• Идиома Pimpl уменьшает время построения приложения, снижая зависимости компиляции между клиентами и реализациями классов.
• Для указателей pImpl
типа std::unique_ptr
следует объявлять специальные функции-члены в заголовочном файле, но реализовывать их в файле реализации. Поступайте так, даже если реализации функций по умолчанию являются приемлемыми.
• Приведенный выше совет применим к интеллектуальному указателю std::unique_ptr
, но не к std::shared_ptr
.
Глава 5
Rvalue-ссылки, семантика перемещений и прямая передача
На первый взгляд, семантика перемещения и прямой передачи кажется довольно простой.
• Семантика перемещенияпозволяет компиляторам заменять дорогостоящие операции копирования менее дорогими перемещениями. Так же, как копирующие конструкторы и копирующие операторы присваивания дают вам контроль над тем, что означает копирование объектов, так и перемещающие конструкторы и перемещающие операторы присваивания предоставляют контроль над семантикой перемещения. Семантика перемещения позволяет также создавать типы, которые могут только перемещаться, такие как std::unique_ptr
, std::future
или std::thread
.
• Прямая передачаделает возможным написание шаблонов функций, которые принимают произвольные аргументы и передают их другим функциям так, что целевые функции получают в точности те же аргументы, что и переданные исходным функциям.
Rvаluе-ссылки представляют собой тот клей, который соединяет две эти довольно разные возможности. Это базовый механизм языка программирования, который делает возможными как семантику перемещения, так и прямую передачу.
С ростом опыта работы с этими возможностями вы все больше понимаете, что ваше первоначальное впечатление было основано только на пресловутой вершине айсберга. Мир семантики перемещения, прямой передачи и rvalue-ccылoк имеет больше нюансов, чем кажется на первый взгляд. Например, std::move
ничего не перемещает, а прямая передача оказывается не совсем прямой. Перемещающие операции не всегда дешевле копирования, а когда и дешевле, то не всегда настолько, как вы думаете; кроме того, они не всегда вызываются в контексте, где перемещение является корректным. Конструкция type&&
не всегда представляет rvalue-ссылку.
Независимо от того, как глубоко вы закопались в эти возможности, может показаться, что можно долго копать еще глубже. К счастью, эта глубина не безгранична. Эта глава доведет вас до коренной породы. Когда вы докопаетесь до нее, эта часть С++11 будет выглядеть намного более осмысленной. Например, вы познакомитесь с соглашениями по использованию std::move
и std::forward
. Вы почувствуете себя намного более комфортно, сталкиваясь с неоднозначной природой type&&
. Вы поймете причины удивительно разнообразного поведения перемещающих операций. Все фрагменты мозаики встанут на свои места. В этот момент вы окажетесь там, откуда начинали, потому что семантика перемещений, прямая передача и rvalue-ссылки вновь покажутся вам достаточно простыми. Но на этот раз они будут оставаться для вас такими навсегда.
В этой главе особенно важно всегда иметь в виду, что параметр всегда является lvalue, даже если его тип — rvalue-ссылка. Иными словами, в фрагменте
void f( Widget&& w);
параметр w
представляет собой lvalue, несмотря на то что его тип — rvalue-ссылка на Widget
. (Если это вас удивляет, вернитесь к обзору lvalue и rvalue, который содержится во введении.)
5.1. Азы std::move
и std::forward
Полезно подойти к std::move
и std::forward
с точки зрения того, чего они не делают . std::move
ничего не перемещает. std::forward
ничего не передает. Во время выполнения они не делают вообще ничего. Они не генерируют выполнимый код — ни одного байта.
std::move
и std::forward
являются всего лишь функциями (на самом деле — шаблонами функций), которые выполняют приведения. std::move
выполняет безусловное приведение своего аргумента к rvalue, в то время как std::forward
выполняет приведение только при соблюдении определенных условий. Это все. Пояснения приводят к новому множеству вопросов, но, по сути, история на этом завершена.
Чтобы сделать историю более конкретной, рассмотрим пример реализации std::move
в C++11. Она не полностью соответствует деталям стандарта, но очень близка к этому.
Интервал:
Закладка: