Улучшенный метод сортировки. QUICK – сортировка
Все рассмотренные выше методы сортировки допускают определенное улучшение. Однако наибольший эффект достигается при модификации наиболее быстрого из всех известных методов - метода прямого обмена. Эта модификация получила название QUICK - сортировка. Суть метода - в середине массива выбирается некоторый граничный элемент, разбивающий весь массив на левую и правую части. Производится одновременное движение слева и справа; если слева находится элемент, больший граничного, то они меняются местами и поиск таких элементов продолжается дальше до границы. Подобная операция повторяется отдельно с левой и правой частями и т.д. Эта идея приводит к тому, что нужно построить процедуру, которая допускает обращение самой к себе, т.е. рекурсивную процедуру, имеющую входными параметрами номера элеметов: 1-ый - левый элемент: L, 2-ой - правый элемент: R. На начало процедуры эти параметры должны получить значения: L = 1; R = N. В процедуре используется цикл REPEAT по сближению левой и правой границ. procedure QUICKSORT (L, R: integer; var M: MAS); var I,J,W,X: integer; begin ¦ {Установка границ, от которых идет движение к середине} ¦ I:= L; J:= R; ¦ {Выбор граничного элемента} X:= M[(L+R) div 2]; ¦ repeat ¦ ¦ { Поиск слева элемента, большего X } ¦ ¦ while X > M[I] do I:= I+1; ¦ ¦ { Поиск справа элемента, меньшего X } ¦ ¦ while X < M[J] do J:= J-1; ¦ ¦ if I <= J then ¦ ¦ begin ¦ ¦ ¦ W:= M[I]; {Обмен местами ¦ ¦ ¦ M[I]:= M[J]; I-го и J-го ¦ ¦ ¦ M[J]:= W; элементов, ¦ ¦ ¦ I:=I+1; дальнейшее продвижение ¦ ¦ ¦ J:=J-1; вперед (назад)} ¦ ¦ end; ¦ ¦ {Выход из цикла, если левый край перевалил за правый} ¦ until I > J; ¦ { Повтор обмена для левой части } ¦ if L < J then QUICKSORT (L,J,M); ¦ { Повтор обмена для правой части } ¦ if R > i then QUICKSORT (I,R,M); ¦ end; ПОЯСНЕНИE. Рассмотрим работу этой процедуры на примере сортировки следующего массива:
Здесь имеем значения L=1; I=1; J=7; R=7. Цикл WHILE X > M[I] при Х = 2 дает сразу же отказ и значение I остается старым, т.е. I = 1. Цикл WHILE X < M[J] при Х = 2 дает J = 6. Сравнение IF I <= J дает 1 <= 6, т.е. истина, отсюда идет обмен между I-м и J-м элементами - первый элемент 6 меняется местом с шестым элементом 1:
и индекс I (J) увеличивается (уменьшается): I:=I+1, т.е. I = 2; J:=J-1, т.е. J = 5. Сравнение I > J, т.е. 2 > 5, является ложным, поэтому идет возврат наверх. Циклы WHILE доводят значения переменной I до 2, а значение J становится равным 4. Так как I=2 <= J=4, то идет обмен между 2-м и 4-м элементами:
и индексы получают значения: I = 3, J = 3. При таких значениях индексов имеем M[I]=M[J]=3, что в результате работы циклов WHILE дает I=3 и J=2. Сравнение I<=J будет ложным, поэтому обмена элементов нет, кроме того, становится истинным условие проверки окончания работы цикла REPEATE (I>J) и происходит переход на рекурсивное обращение к самой процедуре. Из двух условий истинным является первое (сравнение L < J дает 1< 2), поэтому идет обращение к процедуре при параметрах L=1 и R=2. Однако для этих праметров упорядочивание уже произошло. Затем формируется отрезок [3,7], где происходит обмен между 5-м и 7-м элементами:
В результате этой работы массив уже упорядочен, однако затем формируются отрезки [3,6] и [5,6], при которых никаких обменов не происходит, но параметры I, J получают значения: I=6, J=4. Ни одно из сравнений L<J (5<4) и R>I (6>6) не является истинным, поэтому рекурсия завершается и процедура заканчивает свою работу. ЗАМЕЧАНИЕ. Данная процедура очень медленно обрабатывает уже упорядоченный массив. СТРУКТУРЫ ПОСЛЕДОВАТЕЛЬНОГО ДОСТУПА. ЛИНЕЙНЫЕ СПИСКИ
Наиболее простыми динамическими структурами данных являются линейные списки. Мы уже познакомились ранее с примером такого списка: цепочка. Существуют цепочки с нулевым звеном и без него. Характерной особенностью цепочки является то, что при ее формировании очередной элемент всегда записывается в конце, а добавление и исключение элементов производится в любом ее месте. Однако это не всегда удобно для работы, поэтому цепочки организуют специальным образом, в результате чего образуются структуры специального вида: очереди, стеки, деки. Итак, рассмотрим теперь более подробно эти виды динамических структур. Очередь
Очередь - это линейный список, в котором все включения производятся на одном конце списка, а все исключения (и обычно всякий доступ) - на другом. Очередь иногда называют циклической памятью или списком типа FIFO (от первых букв английской фразы "First Input First Output"): первым включается - первым исключается. При работе с очередью мы говорим о ее начале и конце – объекты вставляются в конце очереди и удаляются в начале:
Для характеристики работы с очередью необходимо рассмотреть процедуры ее формирования, добавления, исключения элементов. Условимся в дальнейшем, что будем составлять линейные списки, элементами которых будут числа типа INTEGER. Очевидно, что для организации этих данных необходимо задать описание: type SS = ^ZVENO; ZVENO = record elem: integer; next: SS; end. Рассмотрим сначала алгоритм формирования очереди. Для этого введем три указателя - на начало очереди, на ее конец и текущий указатель: VAR L: SS; {начало очереди} R: SS; {конец очереди} K: SS; {рабочий указатель} el1,el2: integer;{рабочие элементы} Алгоритм формирования очереди представлен на следующей схеме:
Запишем теперь полностью процедуру формирования очереди: procedure FORMIR_OTCHERED_1 (var L, R: SS); var K: SS; EL: integer; begin ¦ { Формирование первого звена очереди } ¦ randomize; EL:= random(10); ¦ new(K); L:= K; R:= K; ¦ K^.elem:= EL; K^.next:= nil; ¦ EL:= random(10); ¦ { Помещение очередного элемента в очередь } ¦ while EL <> 0 do ¦ begin ¦ ¦ new(K); ¦ ¦ K^.elem:= EL; K^.next:= nil; ¦ ¦ R^.next:= K; R:= K; ¦ ¦ EL:= random(10); ¦ end; end. ЗАМЕЧАНИЕ. Из программы видно, что L всегда хранит начало очереди, а R - ее конец. Процедура имеет два возвращаемых параметра, которые идентифицируют получаемую с ее помощью очередь. Первый параметр L позволяет начать просмотр очереди или удалить из нее элемент, а второй R - добавить новый элемент (согласно правилу работы с очередями). Заметим также, что эта процедура формирует очередь из однозначных чисел и признаком конца очереди является число 0. Она предполагает формирование пустой очереди, состоящей из одного нулевого звена:
Если пустой очередью считать очередь без единого звена, то процедура принимает вид: procedure FORMIR_OTCHERED_2 (var L, R: SS); var K: SS; EL1, EL2: integer; begin ¦ {Формирование первого звена очереди } ¦ randomize; EL1:= random(10); ¦ if EL1= 0 then begin L:= nil; R:= L end ¦ else begin new(K); ¦ L:= K; R:= K; K^.next:= nil; ¦ K^.elem:= EL1; ¦ { Помещение очередного элемента в очередь } ¦ EL2:=random(10); ¦ while EL2<>0 do ¦ begin ¦ new(K); ¦ K^.elem:= EL2; K^.next:= nil; ¦ R^.next:= K; R:= K; EL2:= random(10); ¦ end; end; end. Одним из приемов работы с очередями является доступ к ее элементу, в частности, сравнение элемента с каким-то числом или вывод его на печать. Исходя из идеологии построения очереди видно, что выборка любого элемента, как и в файле последовательного доступа, возможна только путем просмотра всех элементов очереди, начиная с ее начала. Это легко сделать, зная ссылки на начало и конец очереди. Наличие двух ссылок очень удобно для просмотра очереди (поиска нужного элемента или вывода его на печать). Действительно, в этом случае достаточно организовать цикл с изменением некоторой ссылочной переменной от значения L до значения R. Таким образом, если необходимо обработать очередь, то следует указать для нее две переменные, где хранятся ссылки на начало и конец. Эти переменные берутся либо непосредственно из программы формирования очереди, либо как выходные параметры процедуры формирования, рассмотренной выше. Ниже следует процедура распечатки элементов очереди, сформированной процедурой пункта 14.1.1: procedure VIVOD_OTCHERED (var L, R: SS); var K: SS; begin ¦ if (L^.elem= 0) or (L= nil) then ¦ writeln('Очеpедь пуста ! ') ¦ else begin ¦ ¦ K:= L; ¦ ¦ write('Элементы очереди: '); ¦ ¦ while K <> R^.next do ¦ ¦ begin ¦ ¦ ¦ write (K^.elem, ' '); ¦ ¦ ¦ K:= K^.next; ¦ ¦ end; ¦ end; end. ЗАМЕЧАНИЕ. В данной процедуре знание ссылки R на конец очереди совсем не обязательно. Здесь можно обойтись только ссылкой на начало, а в цикле WHILE в качестве условия взять сравнение значения переменной K с NIL: WHILE K <> NIL. Добавление нового звена EL к очереди происходит справа (используется указатель R). Рассмотрим сначала алгоритм:
Запишем теперь процедуру добавления звена к очереди: procedure DOBAV_OTCHERED (EL:integer; var L, R: SS); var K: SS; begin ¦ if L^.elem = 0 then R^.elem:= EL ¦ else if L = nil then ¦ begin ¦ ¦ new(K);L:= K; ¦ ¦ R:= K; ¦ ¦ K^.next:= nil; ¦ ¦ K^.elem:= EL ¦ end ¦ else begin ¦ ¦ new(K); ¦ ¦ K^.elem:=el; ¦ ¦ K^.next:=nil; ¦ ¦ R^.next:=K; ¦ ¦ R:=K ¦ end; end. ЗАМЕЧАНИЕ. В данной процедуре сначала проверяется, является ли очередь пустой. Если пустая очередь имеет нулевое звено,то оно заполняется элементом EL. Если же она не содержит звеньев, то создается одно звено по тем же правилам, как при формировании очереди. В общем случае к последнему звену очереди добавляется новое звено. Исключение звена из очереди происходит слева – используется указатель L - и осуществляется одним оператором: L:=L^.next. При такой операции, однако, память не освобождается. Для ее освобождения необходимо дополнительно использовать процедуру DISPOSE.
procedure UDALENIE_OTCHERED (var l, r:ss); begin ¦ if l=nil then writeln('Очеpедь пуста !') ¦ else l:=l^.next end. ОБЩЕЕ ЗАМЕЧАНИЕ. В рассмотренных процедурах признаком конца очереди являлось число 0. Если очередь заполняется символами, то для этого нужно выбрать свой признак конца, например, ".". Для ввода символов, как и для ввода чисел, также можно использовать датчик случайных чисел. Но в этом случае он должен генерировать коды ASCII, которые затем с помощью функции преобразования типов CHR трансформировать в сами символы. Можно элементы очереди вводить с клавиатуры с помощью операторов READLN и READ. При использовании READLN необходимо производить нажатие клавиши ENTER после набора каждого элемента (числа или символа). Оператор же READ позволяет ввести сразу все элементы, заканчивая их набор числом 0 или точкой как признаком конца, причем числа при этом надо отделять друг от друга пробелом. В этом случае для очистки буфера клавиатуры рекомендуется в конце ввода предусмотреть пустой оператор READLN. Заметим, кстати, что все сказанное о формировании очереди распространяется и на другие типы линейных списков. Стек
Стек - это структура данных, которая обеспечивает доступ к списку по принципу LIFO (от первых букв английской фразы "Last Input First Output"): последним вошел, первым вышел. Компонента извлекается из стека таким образом, что первой выбирается та, которая была помещена последней. В стеке доступна только одна позиция - его вершина, где находится последний по времени занесения элемент. Стек, как и очередь, можно представить в виде динамической цепочки звеньев, где первое звено является вершиной стека. Таким образом, в цепочке, отображающей стек, заглавное звено становится излишним. Значением указателя, представляющего стек как единый объект, является ссылка на ВЕРШИНУ стека. Последнее звено цепочки – стека содержит в поле ссылок значение NIL.
Если стек пуст, то значением указателя является ссылка NIL. Перед началом заполнения стека его необходимо сделать пустым, т.е. выполнить "обнуление" указателя стека: STACK:= NIL. Над стеком, как и над очередью, допустимы следующие операции: 1) формирование; 2) занесение нового элемента; 3) удаление элемента; 4) доступ (только для просмотра) к N-му звену стека. Заметим, кстати, что занесение и удаление происходят в стеке исключительно в его вершине.
Запишем теперь саму процедуру формирования стека: procedure SOZDAN_STACK (var ST: SS); var K: SS; EL: integer; begin randomize; EL:= random(10); new(ST); ST:= nil; while EL <> 0 do begin ¦ new(K); K^.elem:= EL; ¦ k^.next:= ST; ST:= K; ¦ EL:= random(10); end; end. ЗАМЕЧАНИЕ. Как видно из процедуры, организация очереди и стека отличается только порядком установления связи: предыдущий элемент очереди указывает на следующий, а следующий элемент стека ссылается на предыдущий. Известно, что новый элемент всегда вставляется на первое место - в вершину стека. Отсюда получаем схему вставки звена в стек:
Процедура имеет два параметра: ST - указатель на стек, EL - заносимый элемент: PROCEDURE VSTAVKA_V_STACK(var ST: SS; EL: integer); var K: SS; begin new(K); K^.elem:= EL; K^.next:= ST; ST:= K end. Схематически процесс удаления можно изобразить так:
По схеме видно, что удаляется N-е звено (вершина стека), которое надо запомнить в специальной ячейке SKLAD: procedure UDALENIE_IZ_STACK(var ST: SS; var SKLAD: integer); begin ¦ SKLAD:= ST^.elem; ¦ ST:= ST^.next end. Мы видим, что здесь переменная ST начинает хранить ссылку на N-1 элемент. Данная процедура имеет недостатки: 1) предполагается, что стек заведомо не пуст, иначе программа зависнет; 2) исключаемое звено не уничтожается, т.к. меняется только ссылка в переменной ST. Если таких удалений несколько, то память будет забита "мусором". Поэтому следует исключить элементы не только из списка, но и из памяти. Для этого можно использовать процедуру DISPOSE. Указанные недостатки учтены в следующей процедуре: procedure UDALENIE_MOD(var ST: SS; var SKLAD: integer); var K: SS; begin ¦ if ST = nil then writeln('стек пустой') ¦ else begin ¦ ¦ SKLAD:= ST^.elem; K:=ST; ¦ ¦ ST:= ST^.next; dispose(K); end; end. Здесь мы ввели в употребление вспомогательную ссылочную переменную К, т.к. писать DISPOSE (ST) нельзя, ведь ST содержит ссылку на вершину стека. Для извлечения из стека N-го элемента необходимо поступить так же, как при выборке элемента из файла, - "прокрутить" стек на N-1 позиций и затем извлечь N-й элемент. Для "прокрутки" стека можно воспользоваться процедурой UDALENIE, т.к. удаление в динамических структурах означает не уничтожение звеньев списка, а только передвижение указателя на следующий элемент. Для написания этой процедуры следует уточнить, что под N-м элементом стека следует понимать элемент, отстоящий на N позиций от вершины стека: procedure VIBORKA_IZ_STACKA(var ST: SS; var SKLAD: integer; N: integer); var K: SS; i: integer; begin ¦ K:= ST; ¦ for i:= 1 to N-1 do ¦ UDALENIE_IZ_STACK(K, SKLAD); ¦ SKLAD:= K^.elem; end. Для вывода на печать элементов стека можно воспользоваться процедурой печати для цепочки, т.к. в этом смысле цепочка ничем не отличается от стека. Отметим только, что элементы стека будут выведены в порядке, обратном его заполнению.
Дек
Дек - это двунаправленная очередь, т.е. линейный список, в котором все включения и исключения (и обычно всякий доступ) достигаются на обоих концах списка:
Более точно следует представить так:
Ôîðìèðîâàíèå äåêà Для организации дека в виде динамической цепочки необходимо иметь следующие типы: type SS = ^ZVENO; ZVENO = record ELEM: integer; next: SS; pred: SS; end. Очевидно, что дек обладает большей общностью, чем стек и очередь; он имеет некоторые общие свойства с каждой из этих структур. Существуют деки с ограниченным входом (реестры) и выходом (архивы). В таких деках соответственно включение и исключение допускается только в одном конце. Строго говоря, при работе с деками достаточно иметь ссылку на один любой элемент. Однако для удобства создадим процедуру, при выходе из которой есть ссылки на оба конца дека: procedure FORMIR_DEK_1(var L, R: SS); var Z: SS; EL: integer; begin ¦ new(L); read(L^.elem); ¦ L^.pred:= nil; R:= L; Z:= L; ¦ WHILE R^.elem <> 0 do ¦ begin ¦ ¦ new(R^.next); R:=R^.next; ¦ ¦ read(r^.elem); R^.pred:= Z; Z:= R; ¦ end; ¦ R^.next:= nil; readln; end. ПРИМЕЧАНИЕ. В рассмотренном примере для образования дека используются три ссылочные переменные: L - начало дека, R – конец дека, Z - промежуточное звено. Однако эта программа может быть упрощена за счет использования более сложных ссылок типа R^.NEXT^.ELEM и R^.NEXT^.PRED. Рассмотрим подробно эти объекты. Пусть ссылочная переменная Z указывает на текущее звено дека:
При таких обозначениях динамическая переменная Z^.NEXT^.ELEM означает числовое поле звена 2 (т.е элемент ELi+1), а переменная Z^.NEXT^.PRED - поле PRED звена 2. Учитывая эти соображения, представим программу формирования дека следующим образом: procedure FORMIR_DEK_2(var L, R: SS); begin ¦ randomize; new(L); ¦ L^.elem:= random (10); ¦ L^.pred:= nil; R:= L; ¦ while R^.elem <> 0 do ¦ begin new(R^.next); ¦ ¦ R^.next^.elem:= random(10); ¦ ¦ R^.next^.pred:= R; R:=R^.next ¦ end; R^.next:= nil end. Для дека справедливы те же операции, что для очереди и стека - вставка и удаление звеньев. Для процедуры вставки в дек, как и в любой другой линейный список, необходимы два параметра: X - звено, после которого (перед которым) надо поместить элемент, и Y - вставляемое звено. Схема вставки остается такой же, как, например, в очереди, но в деке после включения нового звена нужно установить две связи – на последующий и предыдущий элементы.
Популярное: Генезис конфликтологии как науки в древней Греции: Для уяснения предыстории конфликтологии существенное значение имеет обращение к античной... Как распознать напряжение: Говоря о мышечном напряжении, мы в первую очередь имеем в виду мускулы, прикрепленные к костям ... ©2015-2024 megaobuchalka.ru Все материалы представленные на сайте исключительно с целью ознакомления читателями и не преследуют коммерческих целей или нарушение авторских прав. (271)
|
Почему 1285321 студент выбрали МегаОбучалку... Система поиска информации Мобильная версия сайта Удобная навигация Нет шокирующей рекламы |