Сортировка попарным слиянием.


Входное множество рассматривается, как последовательность подмножеств, каждое из которых состоит из единственного элемента и, следовательно, является уже упорядоченным. На первом проходе каждые два соседних одно-элементных множества сливаются в одно двух-элементное упорядоченное множество. На втором проходе двух-элементные множества сливаются в 4-элементные упорядоченные множества и т.д. В конце концов мы получаем одно большое упорядоченное множество.

Важнейшей частью алгоритма является слияние двух упорядоченных множеств. Эту часть алгоритма мы опишем строго.

· 1. [Начальные установки]. Определить длины первого и второго исходных множеств - l1 и l2 соответственно. Установить индексы текущих элементов в исходных множествах i1 и i2 в 1. Установить индекс в выходном множестве j=1.

· 2. [Цикл слияния]. Выполнять шаг 3 до тех пор, пока i1<=l1 и i2<=l2.

· 3. [Сравнение]. Сравнить ключ i1-го элемента из 1-го исходного множества с ключом i2-го элемента из второго исходного множества. Если ключ элемента из 1-го множества меньше, то записать i1-ый элемент из 1-го множества на j-ое место в выходное множество и увеличить i1 на 1. Иначе - записать i2-ой элемент из 2-го множества на j-ое место в выходное множество и увеличить i2 на 1. Увеличить j на 1.

· 4. [Вывод остатка]. Если i1<=l1, то переписать часть 1-го исходного множества от i1 до l1 включительно в выходное множество. Иначе - переписать часть 2-го исходного множества от i2 до l2 включительно в выходное множество.

Программный пример 3.17 иллюстрирует сортировку попарным слиянием в ее обменном варианте - выходные множества формируются на месте входных.

{===== Программный пример 3.17 =====}

{ Сортировка слиянием }

Procedure Sort(var a :Seq);

Var i0,j0,i,j,si,sj,k,ke,t,m : integer;

begin

si:=1; { начальный размер одного множества }

while si < N do begin

{ цикл пока одно множество не составит весь массив }

i0:=1; { нач. индекс 1-го множества пары }

while i0 < N do begin

{ цикл пока не пересмотрим весь массив }

j0:=i0+si; { нач. индекс 2-го множества пары }

i:=i0; j:=j0;

{ размер 2-го множества пары может ограничиваться

концом массива }

if si > N-j0+1 then sj:=N-j0+1 else sj:=si;

if sj > 0 then begin

k:=i0; { нач. индекс слитого множества }

while (i < i0+si+sj) and (j a[j] then begin

{ если эл-т 1-го <= элемента 2-го, он остается на

своем месте, но вых.множество расширяется иначе -

освобождается место в вых.множестве и туда заносится

эл-т из 2-го множества }

t:=a[j];

for m:=j-1 downto k do a[m+1]:=a[m];

a[k]:=t;

j:=j+1; { к след. эл-ту во 2-м множестве }

end; { if a[i] > a[j] }

k:=k+1; { вых. множество увеличилось }

i:=i+1; { если был перенос - за счет сдвига, если

не было - за счет перехода эл-та в вых. }

end; { while }

end; { if sj > 0 }

i0:=i0+si*2; { начало следующей пары }

end; { while i0 < N }

si:=si*2; { размер эл-тов пары увеличивается вдвое }

end; { while si < N }

end;

Результаты трассировки примера приведены в таблице 3.11. Для каждого прохода показаны множества, которые на этом проходе сливаются. Обратите внимание на обработку последнего множества, оставшегося без пары.

Таблица 3.11



Дата добавления: 2016-07-22; просмотров: 1506;


Поиск по сайту:

Воспользовавшись поиском можно найти нужную информацию на сайте.

Поделитесь с друзьями:

Считаете данную информацию полезной, тогда расскажите друзьям в соц. сетях.
Poznayka.org - Познайка.Орг - 2016-2024 год. Материал предоставляется для ознакомительных и учебных целей.
Генерация страницы за: 0.009 сек.