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].wfirst[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-ом из двух концов, мы сгенерируем максимум состояний.