Вторник, 19.03.2024, 06:24
Приветствую Вас, Гость |
Меню сайта
Наш опрос
Нужен ли форум на этом сайте?
Всего ответов: 1296
Статистика

Онлайн всего: 3
Гостей: 3
Пользователей: 0

Сортировка бинарными вставками

В данной статье будет обьяснён принцип алгоритма сортировки бинарными вставками и приведен пример реализации.

Сортировку простыми вставками можно немного улучшить:
Øпоиск "подходящего места" в упорядоченной последовательности можно вести более экономичным способом, который называется Двоичный поиск в упорядоченной последовательности.
Он напоминает детскую игру "больше-меньше": после каждого сравнения обрабатываемая последовательность сокращается в два раза.
Пусть, к примеру, нужно найти место для элемента 7 в таком массиве:
[2  4  6  8  10  12  14  16  18]
Найдем средний элемент этой последовательности (10) и сравним с ним семерку. После этого все, что больше 10 (да и саму десятку тоже), можно смело исключить из дальнейшего рассмотрения:
[2  4  6  8]  10  12  14  16  18
Снова возьмем середину в отмеченном куске последовательности, чтобы сравнить ее с семеркой.
Однако здесь нас поджидает небольшая проблема: точной середины у новой последовательности нет, поэтому нужно решить, который из двух центральных элементов станет этой "серединой". От того, к какому краю будет смещаться выбор в таких "симметричных" случаях, зависит окончательная реализация нашего алгоритма.
Давайте договоримся, что новой "серединой" последовательности всегда будет становиться левый центральный элемент. Это соответствует вычислению номера "середины" по формуле
nomer_sred:= (nomer_lev + nomer_prav) div 2
Итак, отсечем половину последовательности:
2  4  [6  8]  10  12  14  16  18
И снова:
2  4  6  [8]  10  12  14  16  18
2  4  6]  [8  10  12  14  16  18
Таким образом, мы нашли в исходной последовательности место, "подходящее" для нового элемента.
Если бы в той же самой последовательности нужно было найти позицию не для семерки, а для девятки, то последовательность границ рассматриваемых промежутков была бы такой:
[2  4  6  8]  10  12  14  16  18
2  4  [6  8]  10  12  14  16  18
2  4  6  [8]  10  12  14  16  18
2  4  6  8]  [10  12  14  16  18
Из приведенных примеров уже видно, что поиск ведется до тех пор, пока левая граница не окажется правее (!) правой границы.
Кроме того, по завершении этого поиска последней левой границей окажется как раз тот элемент, на котором необходимо закончить сдвиг "хвоста" последовательности.
Будет ли такой алгоритм универсальным?
Давайте проверим, что же произойдет, если мы станем искать позицию для единицы:
[2  4  6  8]  10  12  14  16  18
[2]  4  6  8  10  12  14  16  18]
[2  4  6  8  10  12  14  16  18
Как видим, правая граница становится неопределенной - выходит за пределы массива.
Будет ли этот факт иметь какие-либо неприятные последствия?
Очевидно, нет, поскольку нас интересует не правая, а левая граница.
Добавим число 21 в последовательность.
2  4  6  8  10  [12  14  16  18]
2  4  6  8  10  12  14  [16  18]
2  4  6  8  10  12  14  16  [18]
2  4  6  8  10  12  14  16  18][
Кажется, будто все плохо: левая граница вышла за пределы массива; непонятно, что нужно сдвигать...
Вспомним, однако, что в реальности на (N+1)-й позиции как раз и находится вставляемый элемент (21).
Таким образом, если левая граница вышла за рассматриваемый диапазон, получается, что ничего сдвигать не нужно.
Вообще же такие действия выглядят явно лишними, поэтому от них стоит застраховаться, введя одну дополнительную проверку в текст алгоритма.
Фрагмент программы:

   for  i:= 2  to  n  do   
      if  a[i-1]>a[i]  then 
      begin
         x:= a[i];  
         left:= 1;  
        
right:= i-1;  
         repeat    
            sred:= (left+right) div 2;    
            if  a[sred]<x  then  
               left:= sred+1  
               else  right:= sred-1;      
         until  left>right;
         for j:=i-1 downto left  do
            a[j+1]:= a[j];  
         a[left]:= x;
       end;

Эффективность алгоритма.
Теперь на каждом шаге выполняется не N, а log2N проверок, что уже значительно лучше (для примера, сравните 1000 и 10= log21024).
Следовательно, всего будет совершено N* log2N сравнений.
По количеству пересылок алгоритм по-прежнему имеет сложность О(N2).


Форма входа
Поиск
Мы в сети
Реклама
Для того чтобы не видеть рекламу в правом верхнем углу сайта пройдите простую процедуру регистрации
ФОРУМ
У нас наконецто появился форум! Добро пожаловать! Будьте первыми, задайте направление форуму! =)
--- Не стесняемся - заходим на форум! ---