Meet-in-the-middle — различия между версиями
Dronov (обсуждение | вклад) |
Dronov (обсуждение | вклад) |
||
Строка 16: | Строка 16: | ||
// <tex>sum</tex> - массив сумм <tex> a + b </tex>, <tex> cnt </tex> - счетчик массива <tex> sum </tex> | // <tex>sum</tex> - массив сумм <tex> a + b </tex>, <tex> cnt </tex> - счетчик массива <tex> sum </tex> | ||
'''findsum()''' | '''findsum()''' | ||
− | <tex> | + | sum <tex>\leftarrow \emptyset </tex> |
− | '''for''' <tex> | + | '''for''' a = 0 <tex> \rightarrow </tex> N - 1 |
− | '''for''' <tex> | + | '''for''' b = 0 <tex> \rightarrow </tex> N - 1 |
− | + | sum[cnt].s <tex>\leftarrow</tex> A[a] + B[b] | |
− | + | sum[cnt].a <tex>\leftarrow</tex> a | |
− | + | sum[cnt].b <tex>\leftarrow</tex> b | |
− | + | cnt++ | |
− | '''sort''' | + | '''sort'''(sum) |
− | '''for''' <tex> | + | '''for''' c = 0 <tex> \rightarrow </tex> N - 1 |
− | '''for''' <tex> | + | '''for''' d = 0 <tex> \rightarrow </tex> N - 1 |
− | '''if''' сумма | + | '''if''' сумма -(A[c] + A[d]) есть в массив sum |
− | <tex> | + | index <tex> \leftarrow </tex> индекс суммы -(A[c] + A[d]) |
− | '''return''' | + | '''return''' (sum[index].a, sum[index].b, A[c], A[d]) |
'''return''' "No solution" | '''return''' "No solution" | ||
Итоговое время работы <tex> {O(N^2\log{N}}) </tex>. | Итоговое время работы <tex> {O(N^2\log{N}}) </tex>. |
Версия 23:15, 16 декабря 2012
Определение: |
Meet-in-the-middle (Встреча в середине) — это метод решения уравнения вида | , где и , который работает за время .
Meet-in-the-middle разбивает задачу пополам и решает всю задачу через частичный расчет половинок. Он работает следующим образом : переберем все возможные значения
и запишем пару значений в массив, который мы отсортируем по второму элементу (значению функции). Затем будем перебирать всевозможные значения , для каждого из них будем вычислять , которое мы будем искать в нашем отсортированном массиве. Таким образом, время работы нашего алгоритма составляет на сортировку, и на двоичный поиск, что дает в сумме .Содержание
Задача о нахождение четырех чисел с суммой равной нулю
Дан массив целых чисел
. Требуется найти любые 4 числа, сумма которых равна 0 (одинаковые элементы могут быть использованы несколько раз).Например :
. Решением данной задачи является, например, четверка чисел или .Наивный алгоритм заключается в переборе всевозможных комбинаций чисел. Это решение работает за
. Теперь, с помощью Meet-in-the-middle мы можем сократить время работы до .Для этого заметим, что сумму бинарным поиском, есть ли сумма в массиве .
можно записать как . Мы будем хранить все пар сумм в массиве , который мы отсортируем. Далее перебираем все пар сумм и проверяемРеализация
//- массив сумм , - счетчик массива findsum() sum for a = 0 N - 1 for b = 0 N - 1 sum[cnt].s A[a] + B[b] sum[cnt].a a sum[cnt].b b cnt++ sort(sum) 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 - ограничение по весу рюкзака. sn = N / 2, fn = N - sn; for mask = 0 to 2 ** sn - 1 for j = 0 to sn if j-ый бит mask = 1 first[i].w += w[j]; first[i].c += cost[j]; sort(first); for i = 0 to 2 ** sn - 1 if (существует такое подмножество с индексом j, что first[j].w <= first[i].w && first[j].c >= first[i].c) удалим множество с индексом i из массива first for mask = 0 to 2 ** fn - 1 for j = 0 to fn if j-ый бит mask = 1 curw += w[j + sn]; curcost += cost[j + sn]; В массиве first бинарным поиском находим подмножество, с максимальным весом, который не превышает R - curw if (first[index].w <= R - curw && first[index].c + curcost > ans) ans = first[index].c + curcost print ans
Итоговое время работы
.Задача о нахождении кратчайшего расстояния между двумя вершинами в графе
Еще одна задача, решаемая Meet-in-the-middle — это нахождение кратчайшего расстояния между двумя вершинами, зная начальное состояние, конечное состояние и то, что длина оптимального пути не превышает N.
Стандартным подходом для решения данной задачи, является применение алгоритма обхода в ширину. Пусть из каждого состояния у нас есть K переходов, тогда бы мы сгенерировали состояний. Асимптотика данного решения составила бы . Meet-in-the-middle помогает снизить асимптотику до .
Алгоритм решения
1. Сгенерируем bfs-ом все состояния, доступные из начала и конца за
или меньше ходов.2. Найдем состояния, которые достижимы из начала и из конца.
3. Найдем среди них наилучшее по сумме длин путей.
Таким образом, bfs-ом из двух концов, мы сгенерируем максимум состояний.