Скотт Мейерс - Эффективный и современный С++. 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 - читать онлайн бесплатно полную версию (весь текст целиком)
Интервал:
Закладка:
// T выводится как Widget&
Если мы возьмем тип, выведенный для T
(т.e. Widget&
) и используем его для инстанцирования шаблона, то получим следующее:
void func( Widget& &¶m);
Ссылка на ссылку! Но компиляторы не возражают. Из раздела 5.2 мы знаем, что, поскольку универсальная ссылка param инициализируется с помощью lvalue, тип param
должен быть lvalue-ссылкой, но как компилятор получит результат взятия выведенного типа T и подстановки его в шаблон, который представляет собой конечную сигнатуру функции?
void func( Widget¶m);
Ответ заключается в свертывании ссылок (reference collapsing). Да, вам запрещено объявлять ссылки на ссылки, но компиляторы могут создавать их в определенных контекстах, среди которых — инстанцирование шаблонов. Когда компиляторы генерируют ссылки на ссылки, свертывание ссылок определяет, что будет дальше.
Существуют два вида ссылок (lvalue и rvalue), так что имеются четыре возможные комбинации “ссылка на ссылку” (lvalue на lvalue, lvalue на rvalue, rvalue на lvalue и rvalue на rvalue). Если ссылка на ссылку возникает в контексте, где это разрешено (например, во время инстанцирования шаблона), то ссылки сворачиваются в единственную ссылку согласно следующему правилу:
Если любая из ссылок является lvalue-ссылкой, результат представляет собой lvalue-ссылку. В противном случае (т.e. когда обе ссылки являются rvalue-ссылками) результат представляет собой rvalue-ссылку.
В нашем приведенном выше примере подстановка выведенного типа Widget&
в шаблон func
дает rvalue-ссылку на lvalue-ссылку, и правило свертки ссылок гласит, что результатом является lvalue-ccылкa.
Свертывание ссылок является ключевой частью механизма, обеспечивающего работу std::forward
. Как пояснялось в разделе 5.3, std::forward
применяется к параметрам, являющимся универсальными ссылками, так что обычно его применение имеет следующий вид:
template
void f(T&& fParam) {
… // Некоторая работа
someFunc( std::forward(fParam)); // Передача fParam в
} // someFunc
Поскольку fParam
представляет собой универсальную ссылку, мы знаем, что параметр типа T
будет кодировать информацию о том, являлся ли переданный f
аргумент (т.e. выражение, использованное для инициализации fParam
) lvalue или rvalue. Работа std::forward
заключается в приведении fParam
(lvalue) к rvalue тогда и только тогда, когда T
гласит, что переданный в f
аргумент был rvalue, т.e. если T
не является ссылочным типом.
Вот как можно реализовать std::forward
, чтобы он выполнял описанные действия:
template // В пространстве имен std
T&& forward(typename
remove_reference::type& param) {
return static_cast(param);
}
Этот код не совсем отвечает стандарту (я опустил несколько деталей интерфейса), но отличия не играют роли для понимания того, как ведет себя std::forward
.
Предположим, что аргумент, переданный f
, является lvalue типа Widget
. Тип T
будет выведен как Widget&
, а вызов std::forward
инстанцирует std::forward
. Подстановка Widget&
в реализацию std::forward
дает следующее:
Widget&&& forward(typename
remove_reference< Widget&>::type& param)
{ return static_cast< Widget&&&>(param); }
Свойство типа std::remove_reference::type
дает Widget
(см. раздел 3.3), так что std::forward
превращается в
Widget& && forward( Widget& param)
{ return static_cast(param); }
К возвращаемому типу и приведению также применяется сворачивание ссылок, и результат представляет собой последнюю версию std::forward
для вызова:
Widget&forward( Widget¶m) // В пространстве
{ return static_cast< Widget&>(param); } // имен std
Как можно видеть, когда в шаблон функции f
передается аргумент lvalue, std::forward
инстанцируется для получения и возврата lvalue-ссылки. Приведение внутри std::forward
не делает ничего, поскольку тип param
уже представляет собой Widget&
, так что приведение его к Widget&
ни на что не влияет. Таким образом, lvalue-аргумент, переданный std::forward
, вернет lvalue-ссылку. По определению lvalue-ссылки являются lvalue, так что передача lvalue в std::forward
приводит к возврату lvalue, как и предполагалось.
Предположим теперь, что передаваемый f
аргумент является rvalue типа Widget
. В этом случае выведенный тип параметра типа T
шаблона f
будет просто Widget
. Вызов std::forward
внутри f
, таким образом, будет представлять собой std::forward
. Подстановка Widget
вместо T
в реализации std::forward
дает следующее:
Widget&& forward(typename
remove_reference< Widget>::type& param)
{ return static_cast< Widget&&>(param); }
Применение std::remove_reference
к типу Widget
, не являющемуся ссылкой, дает тот же тип, что и переданный ( Widget
), так что std::forward
превращается в
Widget&& forward( Widget& param)
{ return static_cast(param); }
Здесь нет ссылок на ссылки, так что нет и свертывания ссылок, и это последняя инстанцированная версия std::forward
для этого вызова.
Так как rvalue-ссылки, возвращаемые из функции, определены как rvalue, в этом случае std::forward
превратит параметр fParam
(lvalue) функции f
в rvalue. Конечным результатом является то, что rvalue-аргумент, переданный функции f
, будет передан функции someFunc
как rvalue, и это именно то, что и должно было произойти.
Наличие в С++14 std::remove_reference_t
делает возможным реализовать std::forward
немного более лаконично:
template // С++14; в
T&& forward( remove_reference_t& param) // пространстве
{ // имен std
return static_cast(param);
}
Свертывание ссылок происходит в четырех контекстах. Первый и наиболее распространенный — инстанцирование шаблонов. Второй — генерация типов для переменных auto
. Детали, по сути, те же, что и для шаблонов, поскольку вывод типа для auto
-переменных, по сути, совпадает с выводом типов для шаблонов (см. раздел 1.2). Рассмотрим еще раз пример, приводившийся ранее в данном разделе:
template void func(T&& param);
Widget widgetFactory(); // Функция, возвращающая rvalue
Widget w; // Переменная (lvalue)
func(w); // Вызов функции с lvalue; тип T
// представляет собой Widget&
func(widgetFactory()); // Вызов функции с rvalue; тип T
// представляет собой Widget
Это можно имитировать в виде auto
. Объявление
auto&&w1 = w;
инициализирует w1
с помощью lvalue, выводя, таким образом, для auto
тип Widget&
. Подстановка Widget&
вместо auto
в объявление для w1
дает код со ссылкой на ссылку
Widget& &&w1 = w;
который после сворачивания ссылок принимает вид
Widget&w1 = w;
Интервал:
Закладка: