Meet-in-the-middle — различия между версиями
 (→Реализация:  опечатка)  | 
				Amoniy (обсуждение | вклад)   | 
				||
| Строка 16: | Строка 16: | ||
=== Реализация ===  | === Реализация ===  | ||
   // sum - массив сумм a + b, cnt - счетчик массива sum  |    // sum - массив сумм a + b, cnt - счетчик массива sum  | ||
| − |    '''findsum'''  | + |    '''function''' findsum('''int[]''' A):    | 
     '''for''' a = 0..N - 1  |      '''for''' a = 0..N - 1  | ||
       '''for''' b = 0..N - 1  |        '''for''' b = 0..N - 1  | ||
| − |          sum[cnt].res = A[a] +   | + |          sum[cnt].res = A[a] + A[b]  | 
         sum[cnt].a = a  |          sum[cnt].a = a  | ||
         sum[cnt].b = b  |          sum[cnt].b = b  | ||
| Строка 26: | Строка 26: | ||
     '''for''' c = 0..N - 1  |      '''for''' c = 0..N - 1  | ||
       '''for''' d = 0..N - 1  |        '''for''' d = 0..N - 1  | ||
| − |          '''if''' сумма -(A[c] + A[d]) есть в массив sum  | + |          '''if''' сумма - (A[c] + A[d]) есть в массив sum  | 
            index = индекс суммы -(A[c] + A[d])  |             index = индекс суммы -(A[c] + A[d])  | ||
            '''return''' (sum[index].a, sum[index].b, A[c], A[d])  |             '''return''' (sum[index].a, sum[index].b, A[c], A[d])  | ||
| Строка 44: | Строка 44: | ||
Реализуем данный алгоритм:  | Реализуем данный алгоритм:  | ||
   // N - количество всех вещей, w[] - массив весов всех вещей, cost[] - массив стоимостей всех вещей, R - ограничение по весу рюкзака.  |    // N - количество всех вещей, w[] - массив весов всех вещей, cost[] - массив стоимостей всех вещей, R - ограничение по весу рюкзака.  | ||
| − |    '''  | + |    '''function''' knapsack(): '''int'''  | 
     sn = N / 2  |      sn = N / 2  | ||
     fn = N - sn  |      fn = N - sn  | ||
| − |      '''for''' mask = 0..2 ** sn   | + |      '''for''' mask = 0..2 ** sn - 1  | 
       '''for''' j = 0..sn  |        '''for''' j = 0..sn  | ||
         '''if''' j-ый бит mask == 1  |          '''if''' j-ый бит mask == 1  | ||
| − |            first[i].w += w[j]  | + |            first[i].w += w[j]  | 
           first[i].c += cost[j]  |            first[i].c += cost[j]  | ||
| − |      сортируем first по весу  | + |      sort(first, key = "w") // сортируем first по весу  | 
     '''for''' i = 0..2 ** sn - 1  |      '''for''' i = 0..2 ** sn - 1  | ||
       '''if''' существует такое подмножество с индексом j, что first[j].w <tex> \leqslant </tex> first[i].w '''and''' first[j].c <tex> \geqslant </tex> first[i].c  |        '''if''' существует такое подмножество с индексом j, что first[j].w <tex> \leqslant </tex> first[i].w '''and''' first[j].c <tex> \geqslant </tex> first[i].c  | ||
         удалим множество с индексом i из массива first  |          удалим множество с индексом i из массива first  | ||
| − | |||
     '''for''' mask = 0..2 ** fn - 1  |      '''for''' mask = 0..2 ** fn - 1  | ||
       '''for''' j = 0..fn  |        '''for''' j = 0..fn  | ||
| Строка 62: | Строка 61: | ||
           curw += w[j + sn]  |            curw += w[j + sn]  | ||
           curcost += cost[j + sn]  |            curcost += cost[j + sn]  | ||
| − | |||
       index = позиция, найденная бинарным поиском в массиве first, подмножества с максимальным весом, не превыщающим R - curv  |        index = позиция, найденная бинарным поиском в массиве first, подмножества с максимальным весом, не превыщающим R - curv  | ||
       '''if''' first[index].w <tex> \leqslant </tex> R - curw '''and''' first[index].c + curcost <tex> > </tex> ans  |        '''if''' first[index].w <tex> \leqslant </tex> R - curw '''and''' first[index].c + curcost <tex> > </tex> ans  | ||
| Строка 89: | Строка 87: | ||
* [[Целочисленный двоичный поиск]]  | * [[Целочисленный двоичный поиск]]  | ||
| − | ==  | + | ==Источники информации==  | 
*[http://infoarena.ro/blog/meet-in-the-middle Meet-in-the-middle]  | *[http://infoarena.ro/blog/meet-in-the-middle Meet-in-the-middle]  | ||
*[http://g6prog.narod.ru/dpl.ps Лекции по информатике (36 страница)]  | *[http://g6prog.narod.ru/dpl.ps Лекции по информатике (36 страница)]  | ||
Версия 19:20, 3 января 2017
| Определение: | 
| Meet-in-the-middle (Встреча в середине) — это метод решения уравнения вида , где и , который работает за время , где — время построения множества , — время поиска элемента в множестве , удовлетворяющее решению при заданном , или проверка, что такого не существует. | 
Meet-in-the-middle разбивает задачу пополам и решает всю задачу через частичный расчет половинок. Он работает следующим образом: переберем все возможные значения и запишем пару значений в множество. Затем будем перебирать всевозможные значения , для каждого из них будем вычислять , которое мы будем искать в нашем множестве. Если в качестве множества использовать отсортированный массив, а в качестве функции поиска — бинарный поиск, то время работы нашего алгоритма составляет на сортировку, и на двоичный поиск, что дает в сумме .
Содержание
Задача о нахождении четырех чисел с суммой равной нулю
Дан массив целых чисел . Требуется найти любые числа, сумма которых равна (одинаковые элементы могут быть использованы несколько раз).
Например : . Решением данной задачи является, например, четверка чисел или .
Наивный алгоритм заключается в переборе всевозможных комбинаций чисел. Это решение работает за . Теперь, с помощью Meet-in-the-middle мы можем сократить время работы до .
Для этого заметим, что сумму можно записать как . Мы будем хранить все пар сумм в массиве , который мы отсортируем. Далее перебираем все пар сумм и проверяем бинарным поиском, есть ли сумма в массиве .
Реализация
 // sum - массив сумм a + b, cnt - счетчик массива sum
 function findsum(int[] A): 
   for a = 0..N - 1
     for b = 0..N - 1
       sum[cnt].res = A[a] + A[b]
       sum[cnt].a = a
       sum[cnt].b = b
       cnt++
   sort(sum, key = "res") // сортируем sum по полю res
   for c = 0..N - 1
     for d = 0..N - 1
       if сумма - (A[c] + A[d]) есть в массив sum
          index = индекс суммы -(A[c] + A[d])
          return (sum[index].a, sum[index].b, A[c], A[d])
   return "No solution"
Итоговое время работы .
Если вместо отсортированного массива использовать хэш-таблицу, то задачу можно будет решить за время .
Задача о рюкзаке
Классической задачей является задача о наиболее эффективной упаковке рюкзака. Каждый предмет характеризуется весом ( ) и ценностью (). В рюкзак, ограниченный по весу, необходимо набрать вещей с максимальной суммарной стоимостью. Для ее решения изначальное множество вещей N разбивается на два равных(или примерно равных) подмножества, для которых за приемлемое время можно перебрать все варианты и подсчитать суммарный вес и стоимость, а затем для каждого из них найти группу вещей из первого подмножества с максимальной стоимостью, укладывающуюся в ограничение по весу рюкзака. Сложность алгоритма . Память .
Реализация
Разделим наше множество на две части. Подсчитаем все подмножества из первой части и будем хранить их в массиве . Отсортируем массив по весу. Далее пройдемся по этому массиву и оставим только те подмножества, для которых не существует другого подмножества с меньшим весом и большей стоимостью. Очевидно, что подмножества, для которых существует другое, более легкое и одновременно более ценное подмножество, можно удалять. Таким образом в массиве мы имеем подмножества, отсортированные не только по весу, но и по стоимости. Тогда начнем перебирать все возможные комбинации вещей из второй половины и находить бинарным поиском удовлетворяющие нам подмножества из первой половины, хранящиеся в массиве .
 
Реализуем данный алгоритм:
 // N - количество всех вещей, w[] - массив весов всех вещей, cost[] - массив стоимостей всех вещей, R - ограничение по весу рюкзака.
 function knapsack(): int
   sn = N / 2
   fn = N - sn
   for mask = 0..2 ** sn - 1
     for j = 0..sn
       if j-ый бит mask == 1
         first[i].w += w[j]
         first[i].c += cost[j]
   sort(first, key = "w") // сортируем first по весу
   for i = 0..2 ** sn - 1
     if существует такое подмножество с индексом j, что first[j].w  first[i].w and first[j].c  first[i].c
       удалим множество с индексом i из массива first
   for mask = 0..2 ** fn - 1
     for j = 0..fn
       if j-ый бит mask == 1
         curw += w[j + sn]
         curcost += cost[j + sn]
     index = позиция, найденная бинарным поиском в массиве first, подмножества с максимальным весом, не превыщающим R - curv
     if first[index].w  R - curw and first[index].c + curcost  ans
       ans = first[index].c + curcost
   return ans
Итоговое время работы .
Задача о нахождении кратчайшего расстояния между двумя вершинами в графе
Еще одна задача, решаемая Meet-in-the-middle  —  это нахождение кратчайшего расстояния между двумя вершинами, зная начальное состояние, конечное состояние и то, что длина оптимального пути не превышает .
Стандартным подходом для решения данной задачи, является применение алгоритма обхода в ширину. Пусть из каждого состояния у нас есть  переходов, тогда бы мы сгенерировали  состояний. Асимптотика данного решения составила бы . Meet-in-the-middle помогает снизить асимптотику до . 
Алгоритм решения
1. Сгенерируем bfs-ом все состояния, доступные из начала и конца за или меньше ходов.
2. Найдем состояния, которые достижимы из начала и из конца.
3. Найдем среди них наилучшее по сумме длин путей.
Таким образом, bfs-ом из двух концов, мы сгенерируем максимум  состояний.
