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