Поиск k-ой порядковой статистики за линейное время — различия между версиями

Материал из Викиконспекты
Перейти к: навигация, поиск
Строка 9: Строка 9:
 
Этот алгоритм почти ни чем не отличается от алгоритма [[Поиск k-ой порядковой статистики|поиска k-ой порядковой статистики]], но имеет важное отличие в том, что время работы алгоритма в наихудшем случае равно <tex>O(n)</tex>, что будет доказано ниже. Главная идея алгоритма заключается в том, чтобы ''гарантировать'' хорошее разбиение массива. Алгоритм выбирает такой рассекающий элемент, что количество чисел, которые меньше рассекающего элемента, не менее <tex>\frac{3n}{10}</tex>, где <tex>n</tex> количество элементов в массиве, благодаря этому алгоритм работает за линейной время в любом случае.
 
Этот алгоритм почти ни чем не отличается от алгоритма [[Поиск k-ой порядковой статистики|поиска k-ой порядковой статистики]], но имеет важное отличие в том, что время работы алгоритма в наихудшем случае равно <tex>O(n)</tex>, что будет доказано ниже. Главная идея алгоритма заключается в том, чтобы ''гарантировать'' хорошее разбиение массива. Алгоритм выбирает такой рассекающий элемент, что количество чисел, которые меньше рассекающего элемента, не менее <tex>\frac{3n}{10}</tex>, где <tex>n</tex> количество элементов в массиве, благодаря этому алгоритм работает за линейной время в любом случае.
 
== Описание алгоритма ==
 
== Описание алгоритма ==
#Все <tex>n</tex> элементов входного массива разбиваются на группы по пять элементов, в последней группе будет <tex>n</tex> <tex> mod</tex> <tex> 5</tex> элементов.Эта группа может оказаться пустой при <tex>n</tex> <tex>=</tex> <tex>5</tex>.
+
#Все <tex>n</tex> элементов входного массива разбиваются на группы по пять элементов, в последней группе будет <tex>n</tex> <tex> mod</tex> <tex> 5</tex> элементов.Эта группа может оказаться пустой при <tex>n</tex> кратных <tex>5</tex>.
 
#Сначала сортируется каждая группа, затем выбираем медиану в каждой из этих групп.
 
#Сначала сортируется каждая группа, затем выбираем медиану в каждой из этих групп.
 
#Путем рекурсивного вызова шага 1 определяется медиана <tex>x</tex> из множества медиан, найденных на втором шаге. <tex>x</tex> — рассекающий элемент, <tex>i</tex> — индекс рассекающего элемента.Если медиан окажется четное количество, то на место рассекающего элемента будут претендовать две медианы, переменной <tex>x</tex> будет присвоено значение большей из этих двух медиан.
 
#Путем рекурсивного вызова шага 1 определяется медиана <tex>x</tex> из множества медиан, найденных на втором шаге. <tex>x</tex> — рассекающий элемент, <tex>i</tex> — индекс рассекающего элемента.Если медиан окажется четное количество, то на место рассекающего элемента будут претендовать две медианы, переменной <tex>x</tex> будет присвоено значение большей из этих двух медиан.
Строка 18: Строка 18:
 
     {
 
     {
 
         if (r - l + 1 <= 5) {
 
         if (r - l + 1 <= 5) {
             sort(A[l..r]);
+
             sort(A[l..r]);                           //если элементов не больше 5, сортируем их и возвращаем к-ый элемент
 
             return A[k];
 
             return A[k];
 
         }
 
         }
         for i = l:5:(r - 4)
+
 
 +
         for i = l..(r - 4)
 
             sort(A[i..i + 4];                        //сортируем каждую группу
 
             sort(A[i..i + 4];                        //сортируем каждую группу
         len = r - l + 1;
+
            i += 5;
         Medians[1..n/5] = A[(l + 2):5:(r - 2)];
+
 
 +
         n = r - l + 1;
 +
         Medians[1..n/5] = A[(l + 2):5:(r - 2)];     //создаем массив медиан
 +
        j = l + 2;
 +
        for i = 1..n / 5
 +
            Medians[i] = A[j]
 +
            j += 5;
 +
 
 
         x = select(Medians, 1, n/5, n/10);          //x - рассекающий элемент
 
         x = select(Medians, 1, n/5, n/10);          //x - рассекающий элемент
 +
 
         A = share(A, l, r, x);                      //делим массив относительно элемента x
 
         A = share(A, l, r, x);                      //делим массив относительно элемента x
 +
 
         for i = l to r
 
         for i = l to r
 
             if (A[i] == x)
 
             if (A[i] == x)
                 m = i;                              //находим индекс элемента x в массиве
+
                 m = i;                              //находим индекс элемента x в исходном  массиве
 +
 
 
         if (m = k)
 
         if (m = k)
 
             return A[m];
 
             return A[m];
 
         if (m > k)
 
         if (m > k)
             select(A, k, r, m - k + 1);
+
             select(A, k, r, m - k + 1);             //делаем рекурсивный вызов от той части массива, где находится к-ый элемент
 
         else
 
         else
 
             select(A, l, k, m);
 
             select(A, l, k, m);

Версия 22:00, 27 мая 2012

Определение:
[math]k[/math]-ой порядковой статистикой набора элементов линейно упорядоченного множества называется такой его элемент, который является [math]k[/math]-ым элементом набора в порядке сортировки

Историческая справка

Алгоритм Блюма-Флойда-Пратта-Ривеста-Тарьяна (BFPRT-алгоритм) создан Мануэлем Блюмом (Manuel Blum), Робертом Флойдом (Robert Floyd), Воганом Рональдом Праттом (Vaughan Ronald Pratt), Роном Ривестом (Ron Rivest) и Робертом Тарьяном (Robert Tarjan) в 1973 году.

Идея алгоритма

Этот алгоритм почти ни чем не отличается от алгоритма поиска k-ой порядковой статистики, но имеет важное отличие в том, что время работы алгоритма в наихудшем случае равно [math]O(n)[/math], что будет доказано ниже. Главная идея алгоритма заключается в том, чтобы гарантировать хорошее разбиение массива. Алгоритм выбирает такой рассекающий элемент, что количество чисел, которые меньше рассекающего элемента, не менее [math]\frac{3n}{10}[/math], где [math]n[/math] количество элементов в массиве, благодаря этому алгоритм работает за линейной время в любом случае.

Описание алгоритма

  1. Все [math]n[/math] элементов входного массива разбиваются на группы по пять элементов, в последней группе будет [math]n[/math] [math] mod[/math] [math] 5[/math] элементов.Эта группа может оказаться пустой при [math]n[/math] кратных [math]5[/math].
  2. Сначала сортируется каждая группа, затем выбираем медиану в каждой из этих групп.
  3. Путем рекурсивного вызова шага 1 определяется медиана [math]x[/math] из множества медиан, найденных на втором шаге. [math]x[/math] — рассекающий элемент, [math]i[/math] — индекс рассекающего элемента.Если медиан окажется четное количество, то на место рассекающего элемента будут претендовать две медианы, переменной [math]x[/math] будет присвоено значение большей из этих двух медиан.
  4. Делим массив относительно рассекающего элемента [math]x[/math]. Все элементы меньшие [math]x[/math] будут находиться левее [math]x[/math] в массиве и будут иметь меньший индекс и наоборот, если элементы больше [math]x[/math].
  5. Если [math]i[/math] [math]=[/math] [math]k[/math], то возвращается значение [math]x[/math]. Иначе вызывается рекурсивно шаг 1, и выполняется поиск [math]k[/math]-го в порядке возрастания элемента в левой части массива,если [math]i[/math] [math]\lt [/math] [math]k[/math], или в правой части, если [math]i[/math] [math]\gt [/math] [math]k[/math].

Псевдокод

   select(A, l, r, k)
   {
       if (r - l + 1 <= 5) {
           sort(A[l..r]);                            //если элементов не больше 5, сортируем их и возвращаем к-ый элемент
           return A[k];
       }
       for i = l..(r - 4)
           sort(A[i..i + 4];                        //сортируем каждую группу
           i += 5;
       n = r - l + 1;
       Medians[1..n/5] = A[(l + 2):5:(r - 2)];      //создаем массив медиан
       j = l + 2;
       for i = 1..n / 5
           Medians[i] = A[j]
           j += 5;
       x = select(Medians, 1, n/5, n/10);           //x - рассекающий элемент
       A = share(A, l, r, x);                       //делим массив относительно элемента x
       for i = l to r
           if (A[i] == x)
               m = i;                               //находим индекс элемента x в исходном  массиве
       if (m = k)
           return A[m];
       if (m > k)
           select(A, k, r, m - k + 1);              //делаем рекурсивный вызов от той части массива, где находится к-ый элемент
       else
           select(A, l, k, m);
   }

Пример

На вход подается массив, разобьем элементы на группы по 5 элементов. Отсортируем элементы каждой группы и выберем медианы. Вызовемся рекурсивно от медиан.

BFPRT2.png

Разобьем на группы по 5 медианы. Отсортируем элементы каждой группы и выберем медианы

BFPRT.png

Выберем медианы медиан. В итоге мы получили один элемент равный [math]40[/math]. Это и есть рассекающий элемент.


Анализ времени работы алгоритма

Пусть [math]T(n)[/math] — время работы алгоритма для [math]n[/math] элементов, тогда оно не больше, чем сумма:

  1. времени работы на сортировку групп и разбиение по рассекающему элементу, то есть [math]Cn[/math];
  2. времени работы для поиска медианы медиан, то есть [math]T(\frac{n}{5})[/math];
  3. времени работы для поиска [math]k[/math]-го элемента в одной из двух частей массива, то есть [math]T(s)[/math], где [math]s[/math] — количество элементов в этой части. Но [math]s[/math] не превосходит [math]\frac{7n}{10}[/math], так как чисел, меньших рассекающего элемента, не менее [math]\frac{3n}{10}[/math] — это [math]\frac{n}{10}[/math] медиан, меньших медианы медиан, плюс не менее [math]\frac{2n}{10}[/math] элементов, меньших этих медиан. С другой стороны, чисел, больших рассекающего элемента, так же не менее [math]\frac{3n}{10}[/math], следовательно [math] s \le \frac{7n}{10}[/math], то есть в худшем случае [math] s = \frac{7n}{10}[/math].

Тогда получаем, что [math]T(n) \le T(\frac{n}{5}) + T(\frac{7n}{10}) + Cn [/math]

Покажем, что для всех [math] n [/math] выполняется неравенство [math]T(n) \le 10Cn [/math].

Докажем по индукции:

  1. Очевидно, что для малых [math] n [/math] выполняется неравенство [math]T(n) \le 10Cn [/math]
  2. Тогда, по предположению индукции, [math]T(\frac{n}{5}) \le 10C(\frac{n}{5}) = 2Cn[/math] и [math] T(\frac{7n}{10}) \le 10C(\frac{7n}{10}) = 7Cn[/math], тогда

[math]T(n) \le T(\frac{n}{5}) + T(\frac{7n}{10}) + Cn = 2Cn + 7Cn + Cn = 10Cn \Rightarrow T(n) \le 10Cn[/math]

Так как [math]T(n) \le 10Cn [/math], то время работы алгоритма [math]O(n)[/math]

Литература

  • Кормен, Т., Лейзерсон, Ч., Ривест, Р., Штайн, К. Алгоритмы: построение и анализ

Ссылки