9.0 KiB
Лабораторная работа: Сравнение структур данных для телефонного справочника
1. Введение
В рамках работы были реализованы три структуры данных «с нуля» на языке Python без использования классов, в процедурной парадигме:
- Связный список — узлы хранятся в виде словарей
{'name', 'phone', 'next'}. - Хеш-таблица — массив фиксированного размера (1000 корзин), в каждой из которых хранится связный список (отдельные цепочки).
- Двоичное дерево поиска (BST) — узлы имеют поля
left,right.
Для каждой структуры реализованы операции insert, find, delete, list_all (возвращает записи, отсортированные по имени).
Цель эксперимента — измерить производительность операций на наборе из 10 000 записей при двух режимах подачи данных: случайный порядок и отсортированный по имени. Каждый опыт повторялся 5 раз, результаты усреднены. Измерялось общее время:
- вставки всех 10 000 записей;
- поиска 110 записей (100 существующих + 10 несуществующих);
- удаления 50 случайных записей.
2. Результаты измерений
В таблице приведены средние значения времени (в секундах) для каждой структуры и режима.
| Структура | Режим | Вставка (с) | Поиск (с) | Удаление (с) |
|---|---|---|---|---|
| LinkedList | случайный | 4.1342 | 0.0282 | 0.0138 |
| LinkedList | сортир. | 3.7426 | 0.0248 | 0.0116 |
| HashTable | случайный | 0.00940 | 0.000075 | 0.000037 |
| HashTable | сортир. | 0.00915 | 0.000070 | 0.000032 |
| BST | случайный | 0.02396 | 0.000216 | 0.000126 |
| BST | сортир. | 9.1117 | 0.0796 | 0.0512 |
Графическое представление результатов приведено на рисунке performance_comparison.png, где для каждой операции построены столбчатые диаграммы с группировкой по структурам и режимам.
3. Анализ результатов
3.1. Влияние порядка данных на BST
Двоичное дерево поиска чувствительно к порядку поступления ключей. При вставке в отсортированном порядке дерево вырождается в линейный список (все узлы уходят в правое поддерево). Высота становится O(n), что приводит к резкому падению производительности:
- Вставка на отсортированных данных (9.11 с) медленнее в 380 раз, чем на случайных (0.024 с).
- Поиск замедляется в ~370 раз, удаление — в ~406 раз.
Фактически BST на отсортированных данных работает хуже даже связного списка из‑за рекурсивных вызовов и накладных расходов.
3.2. Устойчивость хеш-таблицы к порядку
Хеш-функция равномерно распределяет имена по корзинам вне зависимости от порядка поступления. Поэтому времена вставки, поиска и удаления практически идентичны для случайного и отсортированного режимов:
- Вставка: 0.00940 с (случ.) vs 0.00915 с (сорт.) — разница в пределах погрешности.
- Поиск и удаление также стабильны.
Средняя сложность O(1) подтверждается на практике.
3.3. Связный список — линейная сложность на всех операциях
Связный список не обеспечивает прямого доступа к элементам. Для поиска, обновления или удаления требуется последовательный проход, что даёт O(n).
- Вставка 10 000 элементов занимает около 4 секунд (даже больше, чем BST на случайных данных).
- Поиск (~0.028 с) на порядок медленнее, чем в хеш-таблице и BST на случайных данных.
- Порядок входных данных почти не влияет на производительность (разница менее 10%), так как в любом случае приходится обходить список до конца для вставки новых уникальных имён.
3.4. Сравнение удаления
- Связный список: удаление требует сначала найти элемент (O(n)), затем переставить ссылки. Время ~0.012–0.014 с, что близко ко времени поиска.
- Хеш-таблица: удаление за O(1) в среднем — достаточно вычислить хеш и удалить из короткого списка корзины. Время ~0.00003–0.00004 с.
- BST: на случайных данных удаление очень быстрое (0.000126 с) благодаря логарифмической высоте. На отсортированных данных время возрастает до 0.051 с (деградация до O(n)).
4. Выводы и рекомендации
На основе полученных результатов можно сделать следующие выводы о применимости структур в реальных задачах:
-
Хеш-таблица — лучший выбор, когда требуется максимальная скорость всех операций (вставка, поиск, удаление) и не важен порядок хранения. Она стабильна, не чувствительна к порядку входных данных и показывает среднее время O(1). Идеальна для реализации словарей, кэшей, индексов по ключу.
-
Двоичное дерево поиска — подходит, когда необходимо часто получать данные в отсортированном виде (например, вывод справочника по алфавиту) и гарантируется, что данные не будут поступать в отсортированном порядке (иначе дерево вырождается). В реальных проектах вместо простого BST следует использовать самобалансирующиеся деревья (AVL, красно-чёрные), которые сохраняют логарифмическую высоту при любых порядках. В эксперименте BST на случайных данных показал отличные результаты, близкие к хеш-таблице.
-
Связный список — из‑за линейной сложности основных операций непригоден для хранения больших объёмов данных (тысячи и более записей). Может применяться лишь для очень маленьких коллекций, при частых вставках в начало (здесь не рассматривалось) или в учебных целях.
Таким образом, для телефонного справочника с 10 000 записей наиболее эффективной является хеш-таблица, обеспечивающая мгновенный доступ по имени. Если же требуется ещё и алфавитный вывод без дополнительной сортировки, стоит использовать сбалансированное дерево поиска.