Реферат: Механизм бектрекинга
Сформулированное правило никак не учитывает события происходящие в вершинах дерева. Между тем вершина от вершины может отличаться и не только положением в дереве. Например, в рассмотренной выше задаче при переходе вниз нарастает сумма, а при переходе вверх та сумма уменьшается. Таким образом, существует класс задач, для которых к дереву комбинаций данных может быть привязана некоторая величина изменяющаяся закономерным образом. Конечно, это не обязательно увеличение. Попробуем описать поведение этой величины в общем виде. Назовём её характеристикой дерева.
Характеристика изменяется внутри некоторого числового интервала.
Это изменение обладает свойством монотонности при движении по дереву вниз.
Существует критическое значение (левая или правая граница интервала), такое, что если характеристика достигает этого критического значения, то дальнейший поиск решения теряет смысл.
С учётом таковой характеристики описанное выше правило обхода дерева немного изменяется и теперь выглядит вот так:
Все ветви уходящие вниз уже пройдены. Тогда необходимо идти по ветви идущей вверх и пометить её как пройденную.
Среди ветвей ведущих вниз есть не пройденные, но характеристика достигла своей критической величины. Тогда необходимо идти по ветви идущей вверх и пометить её как пройденную.
Среди ветвей ведущих вниз есть не пройденные и характеристика не достигла критического значения. Найдём среди них самую левую и пойдём по ней.
Обход дерева комбинаций в соответствии с описанным выше правилом и есть метод бектрекинга (BackTracking - обратный ход). Сформулированное правило достаточно общее и под него подходит довольно много задач, но всё-таки это не самая общая формулировка. Думаю, существует много возможностей расширить правило. Например, можно положить, что характеристик несколько и они зависят друг от друга сложным образом. Можно как-то изменить понятие характеристики. Но мы сейчас заниматься этим не будем, Если есть желание попробуйте самостоятельно сформулировать более широкое правило. А мы сейчас рассмотрим пример применения Бектрекинга.
2. Условие задачи
Некто, назовём его покупателем, зашел в магазин имея на руках определённую сумму денег. Его цель купить как можно большее количество товара (на вес) в пределах имеющейся суммы денег. Для каждого товара задана цена и вес. Именовать товары будем для простоты номером.
Предварительные замечания .
Задача очевидно комбинаторная, поэтому вполне достаточно перебрать все возможные комбинации товаров и для каждой комбинации выяснять суммарный вес и общую стоимость и следить за тем, чтобы общая стоимость на превысила наличную сумму денег.
Из сказанного выше следует, что задачу можно решить обычным полным перебором. Так мы и поступим, а затем в переборное решение добавим механизм бектрекинга.
Нетрудно также заметить, что задача очевидно имеет хорошее рекурсивное решение, но мы будем искать не рекурсивное решение, чтобы не создавать проблем тем, для кого рекурсия быть может новый метод программирования.
3. Решение полным перебором
Думаю, покупатель будет вести себя следующим образом: возьмёт первый товар (а все товары пронумерованы), затем следующий из оставшихся, затем следующий из оставшихся и т.д. пока не заберет все товары.
Затем он выложит последний и снова попытается увеличить количество товаров, но это не имеет смысла, так как последний товар он только что выложил. Тогда он выложит предпоследний и положит в сумку последний, и у него получится новая совокупность товаров.
Действуя, таким образом, покупатель будет получать все новые и новые совокупности и для каждой проверять, а может ли он эту совокупность товаров оплатить и если может, то будет ли она по массе больше той которую он уже запомнил как самую тяжелую.
Теперь то же самое но немного более строго.
Предположим, что у покупателя есть сумка в которой столько же отделов сколько товаров и эти отделы пронумерованы. Каждый раз когда покупатель кладет в сумку новый товар, образуется новая совокупность товаров которая может оказаться искомой. Для того, чтобы это выяснить покупатель после того, как положил товар, взвешивает сумку и выясняет две вещи: может ли он это оплатить и если да, то весит ли сумка больше, нежели запомненный вес. А первоначально сумка не весит ничего.
У покупателя две проблемы:
Большая. Как обеспечить полный перебор всех допустимых совокупностей товаров. Маленькая. Как сделать так, чтобы не было повторов. Ведь может так получится, что он возьмёт одни и те же товары, но в разном порядке. Это конечно не беда, но лишняя работа. Я полагаю, что эти две проблемы решаются следующим образом:
В первом отделе должны побывать все товары. В отделе с номером i должны побывать все товары с номерами на меньшим чем i. Таким образом для каждого отдела с номером i будут получены все допустимые комбинации товаров в остальных отделах с номерами большими чем i и соответственно для отдела с номером 1 будут получены вообще все возможные комбинации.
Наверное стоит привести конкретный пример работ алгоритма. Возьмём следующее множество: 1, 2,3. Для этого множества алгоритм даст следующие комбинации:
(1), (1,2), (1, 2,3), (1, 2,3), (1,3)
(2), (2,3)
(3)
чтобы исключить возможность повтора, перед тем как положить товар в сумку будем проверять нет ли его в сумке уже. Звучит глупо конечно, но дело в том, что реально мы будем работать не в магазине с сумкой, а с массивами и уничтожать элемент массива "магазин", а потом его восстанавливать - это лишняя работа, проще осуществлять названную выше проверку.
program example;
uses crt;