Книга: Эффективное использование STL
Совет 17. Используйте «фокус с перестановкой» для уменьшения емкости
Совет 17. Используйте «фокус с перестановкой» для уменьшения емкости
Предположим, вы пишете программу для нового телешоу «Бешеные деньги». Информация о потенциальных участниках хранится в векторе:
class Contestant {…};
vector<Contestant> contestants;
При объявлении набора участников заявки сыплются градом, и вектор быстро заполняется элементами. Но по мере отбора перспективных кандидатов относительно небольшое количество элементов перемещается в начало вектора (вероятно, вызовом partial_sort
или partition
— см. совет 31), а неудачники удаляются из вектора (как правило, при помощи интервальной формы erase
— см. совет 5). В результате удаления длина вектора уменьшается, но емкость остается прежней. Если в какой-то момент времени вектор содержал данные о 100 000 кандидатов, то его емкость останется равной 100 000, даже если позднее количество элементов уменьшится до 10.
Чтобы вектор не удерживал ненужную память, необходимы средства, которые бы позволяли сократить емкость от максимальной до используемой в настоящий момент. Подобное сокращение емкости обычно называется «сжатием по размеру». Сжатие по размеру легко программируется, однако код — как бы выразиться поделикатнее? — выглядит недостаточно интуитивно. Давайте разберемся, как он работает.
Усечение лишней емкости в векторе contestants производится следующим образом:
vector<Contestant>(contestants).swap(contestants);
Выражение vector<Contestant>(contestants)
создает временный вектор, содержащий копию contestants
; основная работа выполняется копирующим конструктором vector
. Тем не менее, копирующий конструктор vector
выделяет ровно столько памяти, сколько необходимо для хранения копируемых элементов, поэтому временный вектор не содержит лишней емкости. Затем содержимое вектора contestants
меняется местами с временным вектором функцией swap
. После завершения этой операции в contestants
оказывается содержимое временного вектора с усеченной емкостью, а временный вектор содержит «раздутые» данные, ранее находившиеся в contestants
. В этот момент (то есть в конце команды) временный вектор уничтожается, освобождая память, ранее занимаемую вектором contestants
.
Аналогичный прием применяется и по отношению к строкам:
string s; // Создать большую строку и удалить из нее
… // большую часть символов
string(s).swap(s); // Выполнить "сжатие по размеру" с объектом s
Я не могу предоставить стопроцентной гарантии того, что этом прием действительно удалит из контейнера лишнюю емкость. Авторы реализаций при желании могут намеренно выделить в контейнерах vector
и string
лишнюю память, и иногда они так и поступают. Например, контейнер может обладать минимальной емкостью или же значения емкости vector/string
могут ограничиваться степенями 2 (по собственному опыту могу сказать, что подобные аномалии чаще встречаются в реализациях string
, нежели в реализациях vector
. За примерами обращайтесь к совету 15). Таким образом, «сжатие по размеру» следует понимать не как «приведение к минимальной емкости», а как «приведение к минимальной емкости, допускаемой реализацией для текущего размера контейнера». Впрочем, это лучшее, что вы можете сделать (не считая перехода на другую реализацию STL), поэтому «сжатие по размеру» для контейнеров vector
и string
фактически эквивалентно «фокусу с перестановкой».
Кстати говоря, одна из разновидностей «фокуса с перестановкой» может использоваться для очистки контейнера с одновременным сокращением емкости до минимальной величины, поддерживаемой вашей реализацией. Для этого в перестановке используется временный объект vector
или string
, содержимое которого создается конструктором по умолчанию:
vector<Contestant> v;
string s;
… // Использовать v и s
vector<Contestant>().swap(v); // Очистить v с уменьшением емкости
string().swap(s); // Очистить s с уменьшением емкости
Остается сделать последнее замечание, относящееся к функции swap
в целом. Перестановка содержимого двух контейнеров также приводит к перестановке их итераторов, указателей и ссылок. Итераторы, указатели и ссылки, относившиеся к элементам одного контейнера, после вызова swap
остаются действительными и указывают на те же элементы — но в другом контейнере.
- Совет 13. Используйте vector и string вместо динамических массивов
- Совет 14. Используйте reserve для предотвращения лишних операций перераспределения памяти
- Совет 15. Помните о различиях в реализации string
- Совет 16. Научитесь передавать данные vector и string функциям унаследованного интерфейса
- Совет 17. Используйте «фокус с перестановкой» для уменьшения емкости
- Совет 18. Избегайте vector
- Используйте аутсорсинг
- Фокус-группы вместо пудры
- Глава 14 Советы хакера
- 4.15. Советы по конфигурированию Firewall
- 7.7.3. Секреты и советы
- Советы покупателям
- Несколько советов обеспокоенным администраторам
- Совет 48. Всегда включайте нужные заголовки
- 14.12. Короткие советы
- Советы для эффективного рисования
- Советы по навигации в Сети
- Совет 5. Используйте интервальные функции вместо одноэлементных