Вход/Регистрация
Эффективное использование STL
вернуться

Мейерс Скотт

Шрифт:

Для контейнеров

multiset
и
multimap
в качестве возможных кандидатов для поиска первого объекта с заданным значением указаны два алгоритма,
find
и
lower_bound
. Обычно для решения этой задачи выбирается
find
— возможно, вы обратили внимание, что именно этот алгоритм указан в таблице для контейнеров
set
и
map
. Однако
multi
– контейнеры не гарантируют, что при наличии нескольких элементов с заданным значением
find
найдет первый элемент в контейнере; известно лишь то, что будет найден один из этих элементов. Если вы действительно хотите найти первый объект с заданным значением, воспользуйтесь
lower_bound
и выполните вручную вторую часть проверки эквивалентности, описанной в совете 19 (без этой проверки можно обойтись при помощи
equal_range
, но вызов
equal_range
обходится дороже, чем вызов
lower_bound
).

Выбрать между

count
,
find
,
binary_search
,
lower_bound
,
upper_bound
и
equal_range
несложно. Предпочтение отдается тому алгоритму или функции, которые обладают нужными возможностями, обеспечивают нужное быстродействие и требуют минимальных усилий при вызове. Следуйте этой рекомендации (или обращайтесь к таблице), и у вас никогда не будет проблем с выбором.

Совет 46. Передавайте алгоритмам объекты функций вместо функций

Часто говорят, что повышение уровня абстракции языков высокого уровня приводит к снижению эффективности сгенерированного кода. Александр Степанов, изобретатель STL, однажды разработал небольшой комплекс тестов для оценки «платы за абстракцию» при переходе с C на C++. В частности, результаты этих тестов показали, что код, сгенерированный для работы с классом, содержащим

double
, почти всегда уступает по эффективности соответствующему коду, непосредственно работающему с
double
. С учетом сказанного вас может удивить тот факт, что передача алгоритмам объектов функций STL — то есть объектов, маскирующихся под функции, — обычно обеспечивает более эффективный код, чем передача «настоящих» функций.

Предположим, вы хотите отсортировать вектор чисел типа

double
по убыванию. Простейшее решение этой задачи средствами STL основано на использовании алгоритма
sort
с объектом функции типа
greater<double>
:

vector<double> v;

…

sort(v.begin, v.end, greater<double>);

Вспомнив о «плате за абстракцию», программист решает заменить объект функции «настоящей» функцией, которая к тому же оформлена как подставляемая (

inline
):

inline bool doubleGreater(double d1, double d2) {

 return d1 > d2;

}

…

sort(v.begin, v.end, doubleGreater);

Как ни странно, хронометраж двух вызовов sort показывает, что вызов с

greater<double>
почти всегда работает быстрее. В своих тестах я сортировал вектор, содержащий миллион чисел типа
double
, на четырех разных платформах STL с оптимизацией по скорости, и версия с
greater<double>
всегда работала быстрее. В худшем случае выигрыш в скорости составил 50%, в лучшем он достигал 160%. Вот тебе и «плата за абстракцию»…

Факт объясняется просто. Если функция

operator
объекта функции была объявлена подставляемой (явно, с ключевым словом
inline
, или косвенно, посредством определения внутри определения класса), большинство компиляторов благополучно подставляет эту функцию во время создания экземпляра шаблона при вызове алгоритма. В приведенном выше примере это происходит с функцией
greater<double>::operator
. В результате код
sort
не содержит ни одного вызова функций, а для такого кода компилятор может выполнить оптимизацию, недоступную при наличии вызовов (связь между подстановкой функций и оптимизацией компиляторов рассматривается в совете 33 «Effective C++» и главах 8-10 книги «Efficient C++» [10]).

При вызове

sort
с передачей
doubleGreater
ситуация выглядит иначе. Чтобы убедиться в этом, необходимо вспомнить, что передача функции в качестве параметра другой функции невозможна. При попытке передачи функции в качестве параметра компилятор автоматически преобразует функцию в указатель на эту функцию, поэтому при вызове передается указатель. Таким образом, при вызове

sort(v.begin, v.end, doubleGreater);

алгоритму

sort
передается не
doubleGreater
, а указатель на
doubleGreater
. При создании экземпляра шаблона объявление сгенерированной функции выглядит так:

void sort(vector<double>::iterator first, // Начало интервала

 vector<double>:iterator last, // Конец интервала

 bool (*comp)(double, double)); // Функция сравнения

Поскольку

comp
является указателем на функцию, при каждом его использовании внутри sort происходит косвенный вызов функции (то есть вызов через указатель). Большинство компиляторов не пытается подставлять вызовы функций, вызываемых через указатели, даже если функция объявлена с ключевым словом
inline
и оптимизация выглядит очевидной. Почему? Наверное, потому, что разработчики компиляторов не считают нужным ее реализовать. Пожалейте их — народ постоянно чего-нибудь требует, а успеть все невозможно. Впрочем, это вовсе не означает, что требовать не нужно.

  • Читать дальше
  • 1
  • ...
  • 90
  • 91
  • 92
  • 93
  • 94
  • 95
  • 96
  • 97
  • 98
  • 99
  • 100
  • ...

Ебукер (ebooker) – онлайн-библиотека на русском языке. Книги доступны онлайн, без утомительной регистрации. Огромный выбор и удобный дизайн, позволяющий читать без проблем. Добавляйте сайт в закладки! Все произведения загружаются пользователями: если считаете, что ваши авторские права нарушены – используйте форму обратной связи.

Полезные ссылки

  • Моя полка

Контакты

  • chitat.ebooker@gmail.com

Подпишитесь на рассылку: