Скотт Мейерс - Эффективный и современный С++. 42 рекомендации по использованию С++11 и С++14

Тут можно читать онлайн Скотт Мейерс - Эффективный и современный С++. 42 рекомендации по использованию С++11 и С++14 - бесплатно полную версию книги (целиком) без сокращений. Жанр: comp-programming, издательство Вильямс, год 2016. Здесь Вы можете читать полную версию (весь текст) онлайн без регистрации и SMS на сайте лучшей интернет библиотеки ЛибКинг или прочесть краткое содержание (суть), предисловие и аннотацию. Так же сможете купить и скачать торрент в электронном формате fb2, найти и слушать аудиокнигу на русском языке или узнать сколько частей в серии и всего страниц в публикации. Читателям доступно смотреть обложку, картинки, описание и отзывы (комментарии) о произведении.
  • Название:
    Эффективный и современный С++. 42 рекомендации по использованию С++11 и С++14
  • Автор:
  • Жанр:
  • Издательство:
    Вильямс
  • Год:
    2016
  • Город:
    Москва
  • ISBN:
    978-5-8459-2000-3
  • Рейтинг:
    3/5. Голосов: 11
  • Избранное:
    Добавить в избранное
  • Отзывы:
  • Ваша оценка:
    • 60
    • 1
    • 2
    • 3
    • 4
    • 5

Скотт Мейерс - Эффективный и современный С++. 42 рекомендации по использованию С++11 и С++14 краткое содержание

Эффективный и современный С++. 42 рекомендации по использованию С++11 и С++14 - описание и краткое содержание, автор Скотт Мейерс, читайте бесплатно онлайн на сайте электронной библиотеки LibKing.Ru
Эффективный и современный С++
В книге рассматриваются следующие темы. Освоение С++11 и С++14 — это больше, чем просто ознакомление с вводимыми этими стандартами возможностями (например, объявлениями типов
, семантикой перемещения, лямбда-выражениями или поддержкой многопоточности). Вопрос в том, как использовать их эффективно, чтобы создаваемые программы были корректны, эффективны и переносимы, а также чтобы их легко можно было сопровождать. Именно этим вопросам и посвящена данная книга, описывающая создание по-настоящему хорошего программного обеспечения с использованием C++11 и С++14 — т.е. с использованием современного С++.
■ Преимущества и недостатки инициализации с помощью фигурных скобок, спецификации
, прямой передачи и функций
интеллектуальных указателей
■ Связь между
,
, rvalue-ссылками и универсальными ссылками
■ Методы написания понятных, корректных,
лямбда-выражений
■ Чем
отличается от
, как они используются и как соотносятся с API параллельных вычислений С++
■ Какие из лучших методов “старого” программирования на С++ (т.е. С++98) должны быть пересмотрены при работе с современным С++
Более чем 20 лет книги
серии
являются критерием уровня книг по программированию на С++. Понятное пояснение сложного технического материала принесло ему всемирную известность. Он всегда самый желанный гость на международных конференциях, а его услуги консультанта широко востребованы во всем мире.
Скотт Мейерс Эффективный и современный С++, После изучения основ С++ я перешел к изучению того, как применять С++ в промышленном программировании, с помощью серии книг Скотта Мейерса Эффективный С++. Эффективный и современный С++ — наиболее важная из книг серии, предлагающая ключевые рекомендации, стили и идиомы, позволяющие эффективно использовать современный С++. Вы еще не купили эту книгу? Сделайте это прямо сейчас. Герб Саттер,
глава Комитета ISO по стандартизации С++, специалист в области архитектуры программного обеспечения на С++ в Microsoft

Эффективный и современный С++. 42 рекомендации по использованию С++11 и С++14 - читать онлайн бесплатно полную версию (весь текст целиком)

Эффективный и современный С++. 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 &&param); // rvalue-ссылка

Widget &&var1 = Widget(); // rvalue-ссылка

auto &&var2 = var1; // Не rvalue-ссылка

template

void f(std::vector &&param); // rvalue-ссылка

template

void f(T &&param); // Не 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 , так что, когда эта книга готовилась к печати, некоторые члены сообщества С++ начинали именовать универсальные ссылки передаваемыми ссылками . .

Читать дальше
Тёмная тема
Сбросить

Интервал:

Закладка:

Сделать


Скотт Мейерс читать все книги автора по порядку

Скотт Мейерс - все книги автора в одном месте читать по порядку полные версии на сайте онлайн библиотеки LibKing.




Эффективный и современный С++. 42 рекомендации по использованию С++11 и С++14 отзывы


Отзывы читателей о книге Эффективный и современный С++. 42 рекомендации по использованию С++11 и С++14, автор: Скотт Мейерс. Читайте комментарии и мнения людей о произведении.


Понравилась книга? Поделитесь впечатлениями - оставьте Ваш отзыв или расскажите друзьям

Напишите свой комментарий
x