Скотт Мейерс - Эффективный и современный С++. 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 - читать онлайн бесплатно полную версию (весь текст целиком)
Интервал:
Закладка:
void process(Widget&& rvalArg); // Обработка rvalue
template // шаблон, передающий
void logAndProcess( T&& param) // param на обработку
{
auto now = // Получает текущее время
std::chrono::system_clock::now();
makeLogEntry("Bызoв 'process'", now);
process( std::forward(param));
}
Рассмотрим два вызова logAndProcess
, один с lvalue, а другой — с rvalue:
Widget w;
logAndProcess(w); // Вызов с lvalue
logAndProcess(std::move(w)); // Вызов с rvalue
В функции logAndProcess
параметр param
передается функции process
. Функция process
перегружена для lvalue и rvalue. Вызывая logAndProcess
с lvalue, мы, естественно, ожидаем, что lvalue будет передано функции process
как lvalue, а вызывая logAndProcess
с rvalue, мы ожидаем, что будет вызвана перегрузка process
для rvalue.
Однако param
, как и все параметры функций, является lvalue. Каждый вызов process
внутри logAndProcess
будет, таким образом, вызывать перегрузку process
для lvalue. Для предотвращения такого поведения нам нужен механизм для приведения param
к rvalue тогда и только тогда, когда аргумент, которым инициализируется param — аргумент, переданный logAndProcess
, — был rvalue. Именно этим и занимается std::forward
. Вот почему std::forward
представляет собой условное приведение: эта функция выполняет приведение к rvalue только тогда, когда ее аргумент инициализирован rvalue.
Вы можете удивиться, откуда std::forward
может знать, был ли ее аргумент инициализирован rvalue? Например, как в приведенном выше коде std::forward
может сказать, был ли param
инициализирован с помощью lvalue или rvalue? Краткий ответ заключается в том, что эта информация кодируется в параметре T
шаблона logAndProcess
. Этот параметр передается std::forward,
которая восстанавливает закодированную информацию. Детальное описание того, как работает данный механизм, вы найдете в разделе 5.6.
Учитывая, что и std::move
, и std::forward
сводятся к приведению и единственная разница между ними лишь в том, что std::move
всегда выполняет приведение, в то время как std::forward
— только иногда, вы можете спросить, не можем ли мы обойтись без std::move
и просто использовать везде std::forward
. С чисто технической точки зрения ответ утвердительный: std::forward
может сделать все. Необходимости в std::move
нет. Конечно, ни одна из этих функций не является действительно необходимой , потому что мы могли бы просто вручную написать требуемое приведение, но, я надеюсь, мы сойдемся во мнении, что это будет как минимум некрасиво.
Привлекательными сторонами std::move
являются удобство, снижение вероятности ошибок и большая ясность. Рассмотрим класс, в котором мы хотели бы отслеживать количество вызовов перемещающего конструктора. Все, что нам надо, — это счетчик, объявленный как static
, который увеличивался бы при каждом вызове перемещающего конструктора. Полагая, что единственными нестатическими данными класса является std::string
, вот как выглядит обычный (т.e. использующий std::move
) способ реализации перемещающего конструктора:
class Widget {
public:
Widget(Widget&& rhs)
: s( std::move(rhs.s))
{ ++moveCtorCalls; }
…
private:
static std::size_t moveCtorCalls;
std::string s;
};
Чтобы реализовать то же поведение с помощью std::forward
, код должен был бы выглядеть следующим образом:
class Widget {
public:
Widget(Widget&& rhs) // Безусловная,
: s( std::forward(rhs.s)) // нежелательная
{ ++moveCtorCalls; } // реализация
…
};
Заметим сначала, что std::move
требует только аргумент функции ( rhs.s
), в то время как std::forward
требует как аргумент функции ( rhs.s
), так и аргумент типа шаблона ( std::string
). Затем обратим внимание на то, что тип, который мы передаем std::forward
, должен быть не ссылочным, поскольку таково соглашение по кодированию, что передаваемый аргумент является rvalue (см. раздел 5.6). Вместе это означает, что std::move
требует меньшего ввода текста по сравнению с std::forward
и избавляет от проблем передачи типа аргумента, указывающего, что передаваемый аргумент является rvalue. std::move
устраняет также возможность передачи неверного типа (например, std::string&
, что привело бы к тому, что член-данные s
был бы создан с помощью копирования, а не перемещения).
Что еще более важно, так это то, что использование std::move
выполняет безусловное приведение к rvalue, в то время как использование std::forward
означает приведение к rvalue только ссылок, связанных с rvalue. Это два совершенно различных действия. Первое из них обычно настраивает перемещение, в то время как второе просто передает объект другой функции способом, сохраняющим исходную характеристику объекта (lvalue или rvalue). Поскольку эти действия совершенно различны, наличие двух разных функций (и разных имен функций) является преимуществом, позволяющим их различать.
• std::move
выполняет безусловное приведение к rvalue. Сама по себе эта функция не перемещает ничего.
• std::forward
приводит свой аргумент к rvalue только тогда, когда этот аргумент связан с rvalue.
• Ни std::move
, ни std:: forward
не выполняют никаких действий времени выполнения.
5.2. Отличие универсальных ссылок от rvalue-ссылок
Говорят, что истина делает нас свободными, но при соответствующих обстоятельствах хорошо выбранная ложь может оказаться столь же освобождающей. Этот раздел и есть такой ложью. Но поскольку мы имеем дело с программным обеспечением, давайте избегать слова “ложь”, а вместо него говорить, что данный раздел содержит “абстракцию”.
Чтобы объявить rvalue-ссылку на некоторый тип T
, вы пишете T&&
. Таким образом, представляется разумным предположить, что если вы видите в исходном тексте T&&
, то имеете дело с rvalue-ссылками. Увы, не все так просто.
void f(Widget &¶m); // rvalue-ссылка
Widget &&var1 = Widget(); // rvalue-ссылка
auto &&var2 = var1; // Не rvalue-ссылка
template
void f(std::vector &¶m); // rvalue-ссылка
template
void f(T &¶m); // Не rvalue-ссылка
На самом деле “ T&&
” имеет два разных значения. Одно из них — конечно, rvalue-ссылка. Такие ссылки ведут себя именно так, как вы ожидаете: они связываются только с rvalue и их смысл заключается в идентификации объектов, которые могут быть перемещены.
Другое значение “ T&&
” — либо rvalue-ссылка, либо lvalue-ссылка. Такие ссылки выглядят в исходном тексте как rvalue-ссылки (т.e. “ T&&
”), но могут вести себя так, как если бы они были lvalue-ссылками (т.e. “ T&
”). Такая дуальная природа позволяет им быть связанными как с rvalue (подобно rvalue-ссылкам), так и с lvalue (подобно lvalue-ссылкам). Кроме того, они могут быть связаны как с константными, так и с неконстантными объектами, как с объектами volatile
, так и с объектами, не являющимися volatile
, и даже с объектами, одновременно являющимися и const
, и volatile
. Они могут быть связаны практически со всем . Такие беспрецедентно гибкие ссылки заслуживают собственного имени, и я называю их универсальными ссылками [16] В разделе 5.3 поясняется, что к универсальным ссылкам почти всегда может применяться std::forward , так что, когда эта книга готовилась к печати, некоторые члены сообщества С++ начинали именовать универсальные ссылки передаваемыми ссылками .
.
Интервал:
Закладка: