Сортировка вставками
Сортировка вставками (англ. Insertion sort) — алгоритм сортировки, в котором элементы входной последовательности просматриваются по одному, и каждый новый поступивший элемент размещается в подходящее место среди ранее упорядоченных элементов[1]. Вычислительная сложность — .
Сортировка вставками | |
---|---|
| |
Предназначение | Алгоритм сортировки |
Структура данных | Массив |
Худшее время | О(n2) сравнений, обменов |
Лучшее время | O(n) сравнений, 0 обменов |
Среднее время | О(n2) сравнений, обменов |
Затраты памяти | О(n) всего, O(1) вспомогательный |
Медиафайлы на Викискладе |
Описание
На вход алгоритма подаётся последовательность чисел: . Сортируемые числа также называют ключами. Входная последовательность на практике представляется в виде массива с элементами. На выходе алгоритм должен вернуть перестановку исходной последовательности , чтобы выполнялось следующее соотношение [2].
В начальный момент отсортированная последовательность пуста. На каждом шаге алгоритма выбирается один из элементов входных данных и помещается на нужную позицию в уже отсортированной последовательности до тех пор, пока набор входных данных не будет исчерпан. В любой момент времени в отсортированной последовательности элементы удовлетворяют требованиям к выходным данным алгоритма[3].
Данный алгоритм можно ускорить при помощи использования бинарного поиска для нахождения места текущему элементу в отсортированной части. Проблема с долгим сдвигом массива вправо решается при помощи смены указателей[4].
Псевдокод
На вход процедуре сортировки подаётся массив , состоящий из элементов последовательности , которые требуется отсортировать. соответствует — размеру исходного массива. Для сортировки не требуется привлечения дополнительной памяти, кроме постоянной величины для одного элемента, так как выполняется перестановка в пределах массива. В результате работы процедуры во входном массиве оказывается требуемая выходная последовательность элементов[5].
Псевдокод алгоритма:
for j = 2 to A.length do
key = A[j]
i = j-1
while (int i > 0 and A[i] > key) do
A[i + 1] = A[i]
i = i - 1
end while
A[i+1] = key
end [6] |
for i = 2 to n do
x = A[i]
j = i
while (int j > 1 and A[j-1] > x) do
A[j] = A[j-1]
j = j - 1
end while
A[j] = x
end for[7] |
A[0] = - for i = 2 to n do j = i while (j > 0 and A[j] < A[j - 1]) do swap (A[j], A[j - 1]) j = j - 1 end while end for[8][9] |
В последнем варианте обмен x = A[j]; A[j] = A[j-1]; A[j-1] = x
представлен операцией swap из-за чего он немного медленнее. Значение введённого А[0] меньше любого значения остальных элементов[9].
Анализ алгоритма
Время выполнения алгоритма зависит от входных данных: чем большее множество нужно отсортировать, тем большее время потребуется для выполнения сортировки. Также на время выполнения влияет исходная упорядоченность массива. Время работы алгоритма для различных входных данных одинакового размера зависит от элементарных операций, или шагов, которые потребуется выполнить[10].
Для каждой инструкции алгоритма введём временную стоимость и количество повторений, где — количество проверок условия во внутреннем цикле while[11]:
Код | Стоимость | Повторы |
---|---|---|
for j = 2 to A.length | ||
key = A[j] | ||
i = j — 1 | ||
while i > 0 and A[i] > key | ||
A[i+1] = A[i] | ||
i = i — 1 | ||
A[i+1] = key |
Время работы алгоритма сортировки вставками — это сумма времён работы каждого шага[12]: .
Самым благоприятным случаем является отсортированный массив. При этом все внутренние циклы состоят всего из одной итерации, то есть для всех . Тогда время работы алгоритма составит . Время работы линейно зависит от размера входных данных[13].
Анализ наихудшего случая
Наихудшим случаем является массив, отсортированный в порядке, обратном нужному. При этом каждый новый элемент сравнивается со всеми в отсортированной последовательности. Это означает, что все внутренние циклы состоят из j итераций, то есть для всех . Тогда время работы алгоритма составит:
.
Время работы является квадратичной функцией от размера входных данных[14].
Анализ среднего случая
Для анализа среднего случая нужно посчитать среднее число сравнений, необходимых для определения положения очередного элемента. При добавлении нового элемента потребуется, как минимум, одно сравнение, даже если этот элемент оказался в правильной позиции. -й добавляемый элемент может занимать одно из положений. Предполагая случайные входные данные, новый элемент равновероятно может оказаться в любой позиции[15]. Среднее число сравнений для вставки -го элемента[16]:
Для оценки среднего времени работы для n элементов нужно просуммировать[17]:
Временная сложность алгоритма — . Однако, из-за константных множителей и членов более низкого порядка алгоритм с более высоким порядком роста может выполняться для небольших входных данных быстрее, чем алгоритм с более низким порядком роста[18].
Примечания
- Кнут Д. Э. 5.2 Внутренняя сортировка // Искусство программирования. Том 3. Сортировка и поиск = The Art of Computer Programming. Volume 3. Sorting and Searching / под ред. В. Т. Тертышного (гл. 5) и И. В. Красикова (гл. 6). — 2-е изд. — Москва: Вильямс, 2007. — Т. 3. — 832 с. — ISBN 5-8459-0082-1.
- Кормен, 2013, p. 38.
- Кормен, 2013, p. 39.
- Кнут Д. Э. 5.2.1 Сортировка путём вставок // Искусство программирования. Том 3. Сортировка и поиск = The Art of Computer Programming. Volume 3. Sorting and Searching / под ред. В. Т. Тертышного (гл. 5) и И. В. Красикова (гл. 6). — 2-е изд. — Москва: Вильямс, 2007. — Т. 3. — 832 с. — ISBN 5-8459-0082-1.
- Кормен, 2013, p. 39—40.
- Кон, 2013, p. 39—40.
- Н. Вирт. Алгоритмы и структуры данных. — М.: ДМК Пресс, 2010. — С. 74. — 272 с. — ISBN 987-5-94074-584-6.
- Скиена С. Алгоритмы. Руководство по разработке. — 2-е. — СПб.: БХВ-Петербург, 2014. — С. 137. — 720 с. — ISBN 978-5-9775-0560-4.
- Ахо, 2000, p. 237.
- Кормен, 2013, p. 47.
- Кормен, 2013, p. 48.
- Кормен, 2013, p. 48—49.
- Кормен, 2013, p. 49.
- Кормен, 2013, p. 49—50.
- Макконнелл, 2004, p. 74.
- Макконнелл, 2004, p. 75.
- Макконнелл, 2004, p. 75—76.
- Кормен, 2013, p. 51.
Литература
- Ахо А. В., Хопкрофт Д. Э., Ульман Д. Д. Структуры данных и алгоритмы = Data structures and algorithms / Под ред. А. А. Минько. — М.: Вильямс, 2000. — С. 231. — ISBN 5-8459-0122-7.
- Кнут Д. Э. 5.2.1 Сортировка путём вставок // Искусство программирования. Том 3. Сортировка и поиск = The Art of Computer Programming. Volume 3. Sorting and Searching / под ред. В. Т. Тертышного (гл. 5) и И. В. Красикова (гл. 6). — 2-е изд. — Москва: Вильямс, 2007. — Т. 3. — 832 с. — ISBN 5-8459-0082-1.
- Кормен, Т., Лейзерсон, Ч., Ривест, Р., Штайн, К. 2.1. Сортировка вставкой // Алгоритмы: построение и анализ = Introduction to Algorithms / Под ред. И. В. Красикова. — 3-е изд. — М.: Вильямс, 2013. — С. 38—45. — ISBN 5-8459-1794-8.
- Макконнелл Дж. Основы современных алгоритмов = Analysis of Algorithms: An Active Learning Approach / Под ред. С. К. Ландо. — М.: Техносфера, 2004. — С. 72—76. — ISBN 5-94836-005-9.