Скотт Мейерс - Эффективный и современный С++. 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 - читать онлайн бесплатно полную версию (весь текст целиком)
Интервал:
Закладка:
Если вы автор шаблона, противостояние в применении круглых и фигурных скобок может быть особенно неприятным, потому что в общем случае невозможно сказать, какие скобки должны использоваться. Предположим, например, что вы хотите создать объект произвольного типа с произвольным количеством аргументов. Использование шаблонов с переменным количеством параметров позволяет сделать это концептуально достаточно просто:
template
typename... Ts> // Типы используемых аргументов
void doSomeWork(Ts&&... params) {
Создание локального объекта T из params...
}
Есть два способа превратить строку псевдокода в реальный код (см. в разделе 5.3 информацию о std::forward
):
T localObject (std::forward(params)... ); // Круглые скобки
T localObject {std::forward(params)... }; // Фигурные скобки
Рассмотрим следующий вызывающий код:
std::vector v;
…
doSomeWork>(10, 20);
Если doSomeWork
использует при создании объекта localObject
круглые скобки, в результате будет получен std::vector
с 10 элементами. Если же doSomeWork
использует фигурные скобки, то результатом будет std::vector
с двумя элементами. Какой из этих вариантов корректен! Автор doSomeWork
не может этого знать. Это может знать только вызывающий код.
Это именно та проблема, которая встает перед функциями стандартной библиотеки std::make_unique
и std::make_shared
(см. раздел 4.4). Эти функции решают проблему, используя круглые скобки и документируя это решение как части своих интерфейсов [2] Возможен и более гибкий дизайн, который позволяет вызывающим функциям определить, должны ли использоваться круглые или фигурные скобки в функциях, генерируемых из шаблонов. Подробности можно найти в записи от 5 июня 2013 года в Andrzej's С++ blog , “Intuitive interface — Part I”.
.
• Фигурная инициализация является наиболее широко используемым синтаксисом инициализации, предотвращающим сужающие преобразования и нечувствительным к особенностям синтаксического анализа С++.
• В процессе разрешения перегрузки конструкторов фигурные инициализаторы соответствуют параметрам std::initializer_list
, если это возможно, даже если другие конструкторы обеспечивают лучшее соответствие.
• Примером, в котором выбор между круглыми и фигурными скобками приводит к значительно отличающимся результатам, является создание std::vector<���числовой_тип>
с двумя аргументами.
• Выбор между круглыми и фигурными скобками для создания объектов внутри шаблонов может быть очень сложным.
3.2. Предпочитайте nullptr
значениям 0
и NULL
Дело вот в чем: литерал 0
представляет собой int
, а не указатель. Если С++ встретит 0
в контексте, где может использоваться только указатель, он интерпретирует 0
как нулевой указатель, но это — запасной выход. Фундаментальная стратегия С++ состоит в том, что 0
— это значение типа int
, а не указатель.
С практической точки зрения то же самое относится и к NULL
. В случае NULL
имеется некоторая неопределенность в деталях, поскольку реализациям позволено придавать NULL
целочисленный тип, отличный от int
(например, long
). Это не является распространенной практикой, но в действительности не имеет значения, поскольку вопрос не в точном типе NULL
, а в том, что ни 0
, ни NULL
не имеют тип указателя.
В С++98 основным следствием этого факта было то, что перегрузка с использованием типов указателей и целочисленных типов могла вести к сюрпризам. Передача 0
или NULL
таким перегрузкам никогда не приводила к вызову функции с указателем:
void f(int); // Три перегрузки функции f
void f(bool);
void f(void*);
f(0); // Вызов f(int), не f(void*)
f(NULL); // Может не компилироваться, но обычно
// вызывает f(int) и никогда - f(void*)
Неопределенность в отношении поведения f(NULL)
является отражением свободы предоставленной реализациям в отношении типа NULL
. Если NULL
определен, например, как 0L
(т.e. 0
как значение типа long
), то вызов является неоднозначным, поскольку преобразования long
в int
, long
в bool
и 0L
в void*
рассматриваются как одинаково подходящие. Интересно, что этот вызов является противоречием между видимым смыслом исходного текста (“вызываем f
с нулевым указателем NULL
”) и фактическим смыслом (“вызываем f с некоторой разновидностью целых чисел — не указателем”). Это противоречащее интуиции поведение приводит к рекомендации программистам на С++98 избегать перегрузки типов указателей и целочисленных типов. Эта рекомендация остается в силе и в С++11, поскольку, несмотря на рекомендации данного раздела, некоторые разработчики, определенно, продолжат применять 0
и NULL
, несмотря на то что nullptr
является лучшим выбором.
Преимущество nullptr
заключается в том, что это значение не является значением целочисленного типа. Честно говоря, он не имеет и типа указателя, но его можно рассматривать как указатель любого типа. Фактическим типом nullptr
является std::nullptr_t
, ну, а тип std::nullptr_t
циклически определяется как тип значения nullptr
… Тип std::nullptr_t
неявно преобразуется во все типы обычных указателей, и именно это делает nullptr
действующим как указатель всех типов.
Вызов перегруженной функции f
с nullptr
приводит к вызову перегрузки void*
(т.e. перегрузки с указателем), поскольку nullptr
нельзя рассматривать как что-то целочисленное:
f(nullptr); // Вызов f(void*)
Использование nullptr
вместо 0
или NULL
, таким образом, позволяет избежать сюрпризов перегрузки, но это не единственное его преимущество. Оно позволяет также повысить ясность кода, в особенности при применении auto
-переменных. Предположим, например, что у нас есть следующий исходный текст:
auto result = findRecord( /* Аргументы */ );
if ( result == 0) {
…
}
Если вы случайно не знаете (или не можете быстро найти), какой тип возвращает findRecord
, может быть неясно, имеет ли result
тип указателя или целочисленный тип. В конце концов, значение 0
(с которым сравнивается result
) может быть в обоих случаях. С другой стороны, если вы увидите код
auto result = findRecord( /* Аргументы */ );
if ( result == nullptr) {
…
}
то здесь нет никакой неоднозначности: result должен иметь тип указателя.
Особенно ярко сияет nullptr
, когда на сцене появляются шаблоны. Предположим, что у вас есть несколько функций, которые должны вызываться только при блокировке соответствующего мьютекса. Каждая функция получает указатель определенного вида:
Интервал:
Закладка: