Сортировка слиянием — различия между версиями
Кирилл (обсуждение | вклад) |
Кирилл (обсуждение | вклад) |
||
Строка 53: | Строка 53: | ||
(<tex>O(n)</tex> - это время, необходимое на то, чтобы слить два массива). Распишем это соотношение: | (<tex>O(n)</tex> - это время, необходимое на то, чтобы слить два массива). Распишем это соотношение: | ||
− | T(n) = 2T(n/2) + O(n) = 4T(n/4) + 2*O(n) = ... = | + | T(n) = 2T(n/2) + O(n) = 4T(n/4) + 2*O(n) = ... = 2^kT(1) + kO(n). |
− | Осталось оценить <tex>k</tex>. Мы знаем, что <tex> | + | Осталось оценить <tex>k</tex>. Мы знаем, что <tex>2^k=n</tex>, а значит <tex>k=log(n)</tex>. Уравнение примет вид <tex>T(n)=nT(1)+ log(n)O(n)</tex>. Так как <tex>T(1)</tex> - константа, то <tex>T(n)=O(n)+log(n)O(n)=O(nlog(n))</tex>. |
Версия 00:48, 17 мая 2011
Содержание
Сортировка слиянием
Сортировка слиянием — вероятно, один из самых простых алгоритмов сортировки (среди «быстрых» алгоритмов).Сортировка слиянием — стабильный алгоритм сортировки. Это означает, что порядок «равных» элементов не изменяется в результате работы алгоритма. В некоторых задачах это свойство достаточно важно. Этот алгоритм был предложен Джоном фон Нейманом в 1945 году.
Принцип работы
Эта сортировка — хороший пример использования принципа «разделяй и властвуй». Сначала задача разбивается на несколько подзадач меньшего размера. Затем эти задачи решаются с помощью рекурсивного вызова или непосредственно, если их размер достаточно мал. Наконец, их решения комбинируются, и получается решение исходной задачи.
Для решения задачи сортировки эти три этапа выглядят так:
- Сортируемый массив разбивается на две части примерно одинакового размера;
- Каждая из получившихся частей сортируется отдельно, обычно - рекурсивно, тем же самым алгоритмом;
- Два упорядоченных массива половинного размера соединяются в один.
Рекурсивное разбиение задачи на меньшие происходит до тех пор, пока размер массива не достигнет единицы
(любой массив длины 1 можно считать упорядоченным).
Слияние 2-х массивов
Допустим, у нас есть два отсортированных массива А и B размерами
и соответственно, и мы хотим объединить их элементы в один большой отсортированный массив C размером . Для этого можно применить процедуру слияния, суть которой заключается в повторяющемся «отделении» элемента, наименьшего из двух имеющихся в началах исходных массивов, и присоединении этого элемента к концу результирующего массива. Элементы мы переносим до тех пор, пока один из исходных массивов не закончится. После этого оставшийся «хвост» одного из входных массивов дописывается в конец результирующего массива. Пример работы процедуры показан на рисунке:
Алгоритм слияния формально можно записать следующим образом:
a = 0; b = 0; While (a <) and (b < ) If A[a] B[b] C[a + b] = A[a]; a = a + 1; Else C[a + b] = B[b]; b = b + 1; End; End; If a < Copy remain part of A Else Copy remain part of B End;
Рекурсивный алгоритм
Проще всего формализовать этот алгоритм рекурсивным способом. Функция сортирует участок массива от элемента с номером a до элемента с номером b:
Function MergeSort(a,b) If b = a then Exit; c = (a + b)/2; Mergesort(a,c); MergeSort(c + 1,b); Merge fragments (a,c) and (c + 1,b); End
Пример работы алгоритма показан на рисунке:
Время работы
Чтобы оценить время работы этого алгоритма, составим рекуррентное соотношение. Пускай
- время сортировки массива длины n, тогда для сортировки слиянием справедливо ( - это время, необходимое на то, чтобы слить два массива). Распишем это соотношение:T(n) = 2T(n/2) + O(n) = 4T(n/4) + 2*O(n) = ... = 2^kT(1) + kO(n).
Осталось оценить
. Мы знаем, что , а значит . Уравнение примет вид . Так как - константа, то .